| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297 |
- <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN">
- <html>
- <head>
- <meta http-equiv="content-type" content="text/html; charset=utf-8">
- <title></title>
- <meta name="generator" content="LibreOffice 4.2.8.2 (Linux)">
- <meta name="created" content="20150206;163000000000000">
- <meta name="changedby" content="Anatoli Arkhipenko">
- <meta name="changed" content="20161222;100000000000">
- <style type="text/css">
- <!--
- @page { margin: 0.79in }
- p { margin-bottom: 0.08in; direction: ltr; color: #000000; widows: 0; orphans: 0 }
- p.western { font-family: "Liberation Serif", "MS PMincho", serif; font-size: 12pt; so-language: en-US }
- p.cjk { font-family: "WenQuanYi Micro Hei", "MS Mincho"; font-size: 12pt; so-language: zh-CN }
- p.ctl { font-family: "Lohit Hindi", "MS Mincho"; font-size: 12pt; so-language: hi-IN }
- a:link { color: #0000ff }
- -->
- </style>
- </head>
- <body lang="en-US" text="#000000" link="#0000ff" bgcolor="#ffffff" dir="ltr" style="background: #ffffff">
- <p class="western" style="margin-bottom: 0in"><font size="4" style="font-size: 15pt"><b>Task
- Scheduler</b></font></p>
- <p class="western" style="margin-bottom: 0in"><b>Cooperative
- multitasking for Arduino microcontrollers</b></p>
- <p class="western" style="margin-bottom: 0in; border-top: none; border-bottom: 1px solid #000000; border-left: none; border-right: none; padding-top: 0in; padding-bottom: 0.01in; padding-left: 0in; padding-right: 0in">
- <font size="2" style="font-size: 11pt"><b>Version 2.2.1: 2016-12-20</b></font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>OVERVIEW</b>:</p>
- <p class="western" style="margin-bottom: 0in">A lightweight
- implementation of cooperative multitasking (task scheduling)
- supporting:</p>
- <ol>
- <li><p class="western" style="margin-bottom: 0in">Periodic task
- execution, with dynamic execution period in <b>milliseconds
- (</b>default<b>)</b> or <b>microseconds (</b>if explicitly enabled<b>)</b>
- – frequency of execution</p>
- <li><p class="western" style="margin-bottom: 0in">Number of
- iterations (limited or infinite number of iterations)</p>
- <li><p class="western" style="margin-bottom: 0in">Execution of tasks
- in predefined sequence</p>
- <li><p class="western" style="margin-bottom: 0in">Dynamic change of
- task execution parameters (frequency, number of iterations, callback
- methods)</p>
- <li><p class="western" style="margin-bottom: 0in">Power saving via
- entering IDLE sleep mode when tasks are not scheduled to run</p>
- <li><p class="western" style="margin-bottom: 0in">Support for
- event-driven task invocation via Status Request object</p>
- <li><p class="western" style="margin-bottom: 0in">Support for task
- IDs and Control Points for error handling and watchdog timer</p>
- <li><p class="western" style="margin-bottom: 0in">Support for Local
- Task Storage pointer (allowing use of same callback code for
- multiple tasks)</p>
- <li><p class="western" style="margin-bottom: 0in">Support for
- layered task prioritization</p>
- </ol>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Scheduling overhead:
- between 15 and 18 microseconds per scheduling pass (Arduino UNO rev 3
- @ 16MHz clock, single scheduler w/o prioritization)</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>TASK</b>:</p>
- <p class="western" style="margin-bottom: 0in">“Task” is an
- action, a part of the program logic, which requires scheduled
- execution. A concept of Task combines the following aspects:</p>
- <ol>
- <li><p class="western" style="margin-bottom: 0in">Program code
- performing specific activities (callback methods)</p>
- <li><p class="western" style="margin-bottom: 0in">Execution interval</p>
- <li><p class="western" style="margin-bottom: 0in">Number of
- execution iterations</p>
- <li><p class="western" style="margin-bottom: 0in">(Optionally)
- Execution start event (Status Request)</p>
- <li><p class="western" style="margin-bottom: 0in">(Optionally)
- Pointer to a Local Task Storage area</p>
- </ol>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>Tasks</b> perform
- certain functions, which could require periodic or one-time
- execution, update of specific variables, or waiting for specific
- events. Tasks also could be controlling specific hardware, or
- triggered by hardware interrupts.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">For execution purposes
- <b>Tasks</b> are linked into execution <b>chains</b>, which are
- processed by the <b>Scheduler</b> in the order they were added
- (linked together).</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>Starting with
- version 2.0.0 TaskScheduler supports task prioritization. Please
- refer to the specific chapter of this manual for details on layered
- prioritization. </b>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Each task performs its
- function via a callback method. Scheduler calls Task’s callback
- method periodically until task is disabled or runs out of iterations.
- In addition to “regular” callback method, two additional methods
- could be utilized for each task: a callback method invoked every time
- the task is enabled, and a callback method invoked once when the task
- is disabled. Those two special methods allow tasks to properly
- initiate themselves for execution, and clean-up after execution is
- over (E.g., setup pin modes on enable, and always bring pin level to
- LOW at the end).
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Tasks are responsible
- for supporting <b>cooperative</b> <b>multitasking</b> by being “good
- neighbors”, i.e., running their callback methods quickly in a
- non-blocking way, and releasing control back to scheduler as soon as
- possible.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>Scheduler</b> is
- executing Tasks' callback methods in the order the tasks were added
- to the chain, from first to last. Scheduler stops and exists after
- processing the chain once in order to allow other statements in the
- main code of <b>loop()</b> method to run. This is referred to as a
- <b>“scheduling pass”.</b></p>
- <p class="western" style="margin-bottom: 0in">(Normally, there is no
- need to have any other statements in the <b>loop</b>() method other
- than the Scheduler's <b>execute</b>() method).</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>Below is the flowchart of a Task lifecycle:</b></p>
- <p class="western" style="margin-bottom: 0in"><img src="" name="Image1" align="bottom" width="664" height="517" border="0"></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>TaskScheduler</b>
- library maybe compiled with different compilation controls
- enabled/disabled. This is a way to limit TaskScheduler functionality
- (and size) for specific purpose (sketch). This is achieved by
- defining specific #<b>define</b> parameters <i>before</i>
- TaskScheduler.h header file. Specifically:</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">If compiled with
- <font face="Courier New, monospace">_TASK_SLEEP_ON_IDLE_RUN</font>
- enabled, the scheduler will place processor into IDLE sleep mode (for
- approximately 1 ms, as the timer interrupt will wake it up), after
- what is determined to be an “idle” pass. An Idle Pass is a pass
- through the task chain when no Tasks were scheduled to run their
- callback methods. This is done to avoid repetitive idle passes
- through the chain when no tasks need to be executed. If any of the
- tasks in the chain always requires immediate execution (aInterval =
- 0), then there will be no IDLE sleep between task's callback method
- execution.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>NOTE: </b>Task
- Scheduler uses <b>millis()</b> (or <b>micros()</b>) to determine if
- tasks are ready to be invoked. Therefore, if you put your device to
- any “deep” sleep mode disabling timer interrupts, the
- <b>millis()/micros()</b> count will be suspended, leading to
- effective suspension of scheduling. Upon wake up, active tasks need
- to be re-enabled, which will effectively reset their internal time
- scheduling variables to the new value of <b>millis()/micros(). </b>Time
- spent in deep sleep mode should be considered “frozen”, i.e., if
- a task was scheduled to run in 1 second from now, and device was put
- to sleep for 5 minutes, upon wake up, the task will still be
- scheduled 1 second from the time of wake up. Executing <b>enable()
- </b>method on this tasks will make it run as soon as possible. This
- is a concern only for tasks which are required to run in a truly
- periodical manner (in absolute time terms).
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">In addition to
- time-only (<b>millis()/micros</b>() only) invocation, tasks can be
- scheduled to wait on an event employing StatusRequest objects (more
- about Status Requests later).</p>
- <p class="western" style="margin-bottom: 0in">Consider a scenario
- when one task (t1) is performing a function which affects execution
- of many tasks (t2, t3). In this case the task t1 will “signal”
- completion of its function via Status Request object. Tasks t2 and t3
- are “waiting” on the same Status Request object. As soon as
- status request completes, t2 and t3 are activated.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Alternative scenario is
- the ne task (t1) and waiting for the completion of a number of tasks
- (t2, t3). When done, t2 and t3 signal completion of their functions,
- t1 is invoked.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Please see the code
- examples at the end of this document, and included with the library
- package for details.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>COMPILE PARAMETERS:</b></p>
- <p class="western" style="margin-bottom: 0in">This library could be
- compiled in several configurations.</p>
- <p class="western" style="margin-bottom: 0in">Parameters (<b>#define</b>s)
- defining what functionality should or should not be included need be
- defined before the library header file in the body of Arduino sketch.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">#define <b>_TASK_MICRO_RES</b></p>
- <p class="western" style="margin-bottom: 0in">...will compile the
- library with microsecond scheduling resolution, instead of default
- millisecond resolution.</p>
- <p class="western" style="margin-bottom: 0in">All time parameters for
- execution interval, delay, etc. will be treated as <b>microseconds</b>,
- instead of milliseconds.
- </p>
- <p class="western" style="margin-bottom: 0in"><b>NOTE: </b>Sleep mode
- <b>SLEEP_MODE_IDLE</b> (see below) is automatically <b>disabled</b>
- for microsecond resolution. Time constants <b>TASK_SECOND,
- TASK_MINUTE</b> and <b>TASK_HOUR</b> are adjusted for microsecond
- duration.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">#define
- <b>_TASK_TIMECRITICAL</b></p>
- <p class="western" style="margin-bottom: 0in">...will compile the
- library with time critical tracking option enabled.</p>
- <p class="western" style="margin-bottom: 0in">Time critical option
- keeps track when current execution took place relative to when it was
- scheduled, and where next execution time of the task falls. Two
- methods provide this information.
- </p>
- <p class="western" style="margin-bottom: 0in"><b>Task::getStartDelay()
- </b>method: return number of milliseconds (or microseconds) between
- current system time (millis/micros) and point in time when the task
- was scheduled to start. A value of 0 (zero) indicates that task
- started right on time per schedule.
- </p>
- <p class="western" style="margin-bottom: 0in"><b>Task::getOverrun()</b>
- method: If <b>getOverrun </b>returns a negative value, this Task’s
- next execution time point is <i>already</i> in the past, and task is
- behind schedule. This most probably means that either task’s
- callback method's runtime is too long, or the execution interval is
- too short (and therefore schedule is too aggressive).</p>
- <p class="western" style="margin-bottom: 0in">A positive value
- indicates that task is on schedule, and callback methods have enough
- time to finish before the next scheduled pass.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">#define
- <b>_TASK_SLEEP_ON_IDLE_RUN</b></p>
- <p class="western" style="margin-bottom: 0in">...will compile the
- library with the <b>sleep</b> option enabled (AVR boards only).</p>
- <p class="western" style="margin-bottom: 0in">When enabled, scheduler
- will put the microcontroller into <b>SLEEP_MODE_IDLE</b> state if
- none of the tasks’ callback methods were activated during execution
- pass. <b>IDLE</b> state is interrupted by timers once every 1 ms.
- Putting microcontroller to IDLE state helps conserve power. Device
- in SLEEP_MODE_IDLE wakes up to all hardware and timer interrupts, so
- scheduling is kept current.</p>
- <p class="western" style="margin-bottom: 0in"><b>NOTE: </b>This
- compilation option is not available with the microsecond resolution
- option.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- #define <b>_TASK_STATUS_REQUEST</b></p>
- <p class="western" style="margin-bottom: 0in">…will compile
- TaskScheduler with support for StatusRequest object. Status Requests
- are objects allowing tasks to wait on an event, and signal event
- completion to each other.
- </p>
- <p class="western" style="margin-bottom: 0in"><b>NOTE: </b> tas of
- version 2.2.1 each task has internal StatusRequest object, which
- triggered active at the moment Task is enabled, and triggered
- complete at the moment the task is disabled. These events could be
- used by other Tasks for event-driven execution</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">#define <b>_TASK_WDT_IDS</b></p>
- <p class="western" style="margin-bottom: 0in">…will compile
- TaskScheduler with support for Task IDs and Control Points. Each task
- can be (and is by default) assigned an ID, which could be used to
- identify the task in case there is a problem with it. Furthermore
- within the task, Control Points could be defined to further help with
- pinpointing potential problem areas. For instance, the tasks which
- deal with external resources (sensors, serial communications,
- anything hardware dependent) can be blocked (or hung), by failed
- hardware. In this case, a watchdog timer could be employed to trap
- such a failed task, and identify which one (by task id) and where in
- the task (by a control point) the problem is likely located.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>NOTE: </b>by
- default, talk IDs are assigned sequentially (1, 2, 3, …) to the
- tasks as they are being created. Programmer can assign a specific
- task id. <b>Task ids are unsigned integers.</b></p>
- <p class="western" style="margin-bottom: 0in">Control points provide
- a way to identify potential problem points within a task. Control
- points are <b>unsigned integers </b>as well. Please note that there
- is only one control point per task, and it is set to zero when the
- task’s callback method is invoked (this is done to prevent “stray”
- control point from previous task(s) confusing the matters.</p>
- <p class="western" style="margin-bottom: 0in">Example #7 contains a
- test of task ID and control points functionality.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">#define
- <b>_TASK_LTS_POINTER</b></p>
- <p class="western" style="margin-bottom: 0in">…will compile
- TaskScheduler with support for Local Task Storage pointer (LTS). LTS
- is a generic (void*) pointer which could be set to reference a
- variable or a structure specific to a particular task. A callback
- method can get access to specific variables by getting reference to a
- currently running task from the scheduler, and then casting (void*)
- LTS pointer to the appropriate pointer type.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>NOTE: </b>above
- parameters are<b> DISABLED </b>by default, and need to be explicitly
- enabled by placing appropriate #define statements in front of the
- #include statement for the TaskScheduler header file.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">#define <b>_TASK_PRIORITY</b></p>
- <p class="western" style="margin-bottom: 0in">…will compile
- TaskScheduler with support for layered task prioritization. Task
- prioritization is achieved by creating several schedulers, and
- organizing them in priority layers. Tasks are assigned to schedulers
- corresponding to their priority. Tasks assigned to the “higher”
- layers are evaluated for invocation more frequently, and are given
- priority in execution in case of the scheduling coincidence. More
- about layered prioritization in the API documentation and
- TaskScheduler examples.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>TASK PRIORITY AND
- COOPERATIVE MULTITASKING:</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Starting with version
- 2.0.0 TaskScheduler supports task prioritization. Priority is
- associated with a <b>Scheduler</b>, not individual <b>Tasks</b>,
- hence the concept of priority layers. Tasks subsequently are assigned
- to schedulers corresponding to their desired priority. The lowest
- priority Scheduler is called “<b>base scheduler</b>” or “<b>base
- layer</b>”. Let’s call higher priority schedulers by their
- priority number, with larger number corresponding to higher priority
- of task execution.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Task prioritization is
- achieved by executing the entire chain of tasks of the higher
- priority scheduler for every single step (task) of the lower priority
- chain. <b>Note</b> that actual callback method invocation depends on
- priority <b>and </b>the timing of task schedule. However, higher
- priority tasks are evaluated more frequently and are given priority
- in case of scheduling collision.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">For most tasks
- TaskScheduler <b>does not need </b>task priority functionality.
- Prioritization requires additional scheduling overhead, and should be
- used only for critical tasks.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">A few points on that:</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">1. Plain (non-layered)
- execution chain is simple and efficient. The main idea is to minimize
- scheduling overhead by Scheduler going through the chain. Each
- priority layer adds scheduling overhead to overall task chain
- execution. Let’s review 3 scenarios:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>I.
- Flat chain of 7 tasks:</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Scheduling
- evaluation sequence:</p>
- <p class="western" align="center" style="margin-left: 0.49in; margin-bottom: 0in">
- 1 <font face="Wingdings"></font> 2 <font face="Wingdings"></font>
- 3 <font face="Wingdings"></font> 4 <font face="Wingdings"></font>
- 5 <font face="Wingdings"></font> 6 <font face="Wingdings"></font>
- 7</p>
- <p class="western" align="center" style="margin-left: 0.49in; margin-bottom: 0in">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Scheduling
- overhead:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">O
- = B * T = 7 * 18 = 126 microseconds,</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Where:</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">O
- – scheduling overhead</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">B
- – number of tasks in the base layer</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">T
- – scheduling overhead of a single task execution evaluation
- (currently with Arduino Uno running at 16 Mhz is between 15 and 18
- microseconds).</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>II.
- Two priority layers of 7 tasks. </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b> </b>Tasks
- 1, 2, 3, 4, 5 are base priority and 6, 7 are higher priority:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Scheduling
- evaluation sequence:</p>
- <p class="western" align="center" style="margin-left: 0.49in; margin-bottom: 0in">
- 6 <font face="Wingdings"></font> 7 <font face="Wingdings"></font>
- 1 <font face="Wingdings"></font> 6 <font face="Wingdings"></font>
- 7 <font face="Wingdings"></font> 2 <font face="Wingdings"></font>
- 6 <font face="Wingdings"></font> 7 <font face="Wingdings"></font>
- 3 <font face="Wingdings"></font> 6 <font face="Wingdings"></font>
- 7 <font face="Wingdings"></font> 4 <font face="Wingdings"></font>
- 6 <font face="Wingdings"></font> 7 <font face="Wingdings"></font>
- 5</p>
- <p class="western" align="center" style="margin-left: 0.49in; margin-bottom: 0in">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Scheduling
- overhead:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">O
- = (B + B * P1) * T = (5 + 5 * 2) * 18 = 270 microseconds,</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Where:</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">O
- – scheduling overhead</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">B
- – number of tasks in the base layer</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">P1
- – number of tasks in the priority 1 layer</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">T
- – scheduling overhead of a single task execution evaluation
- (currently with Arduino Uno running at 16 Mhz is between 15 and 18
- microseconds).</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>III.
- Three priority layers of 7 tasks. </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b> </b>Tasks
- 1, 2, 3, are base priority, 4, 5 are priority 1, and 6, 7 are
- priority 2:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Scheduling
- evaluation sequence:</p>
- <p class="western" align="center" style="margin-left: 0.49in; margin-bottom: 0in">
- 6 <font face="Wingdings"></font> 7 <font face="Wingdings"></font>
- 4 <font face="Wingdings"></font> 6 <font face="Wingdings"></font>
- 7 <font face="Wingdings"></font> 5 <font face="Wingdings"></font>
- 1 <font face="Wingdings"></font> 6 <font face="Wingdings"></font>
- 7 <font face="Wingdings"></font> 4 <font face="Wingdings"></font>
- 6 <font face="Wingdings"></font> 7 <font face="Wingdings"></font>
- 5 <font face="Wingdings"></font> 2 <font face="Wingdings"></font>6
- <font face="Wingdings"></font> 7 <font face="Wingdings"></font>
- 4 <font face="Wingdings"></font> 6 <font face="Wingdings"></font>
- 7 <font face="Wingdings"></font> 5 <font face="Wingdings"></font>
- 3
- </p>
- <p class="western" align="center" style="margin-left: 0.49in; margin-bottom: 0in">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Scheduling
- overhead:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">O
- = (B + B * P1 + B * P1 * P2) * T = (3 + 3 * 2 + 3 * 2 * 2) * 18 = 378
- microseconds,</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Where:</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">O
- – scheduling overhead</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">B
- – number of tasks in the base layer</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">P1
- – number of tasks in the priority 1 layer</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">P2
- – number of tasks in the priority 2 layer</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">T
- – scheduling overhead of a single task execution evaluation
- (currently with Arduino Uno running at 16 Mhz is between 15 and 18
- microseconds).</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Scheduling overhead of
- a 3 layer prioritization approach is 3 times higher than that of a
- flat execution chain. <b>Do</b> evaluate if task prioritization is
- really required for your sketch.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">2. TaskScheduler is <b>NOT</b>
- a pre-emptive multi-tasking library. Nor is it a Real-Time OS. There
- is no way to break execution of one task in favor of another.
- Therefore callback methods require careful programming for
- cooperative behavior.</p>
- <p class="western" style="margin-bottom: 0in">This has, however,
- significant benefits: you don't need to worry about concurrency
- inside the callback method, since only one callback method runs at a
- time, and could not be interrupted. All resources are yours for that
- period of time, no one can switch the value of variables (except
- interrupt functions of course...), etc. It is a stable and
- predictable environment, and it helps a lot with writing stable code.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">A number of things
- could be done instead of priorities:</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">1. Schedule your
- critical tasks to run more frequently than the other tasks</p>
- <p class="western" style="margin-bottom: 0in">. <br>(Since you can
- control the interval, you could also change the task to run more or
- less frequently as the situation demands).</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">2. If one particular
- callback routine is critical, create a couple of tasks referring to
- the same callback and "sprinkle" them around the chain:</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"> <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">Scheduler
- ts;</font></font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"> <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">Task
- t1(20, TASK_FOREVER, &callback1, &ts);</font></font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"> <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">Task
- t2(1000, TASK_FOREVER, &callback2, &ts);</font></font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"> <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">Task
- t3(20, TASK_FOREVER, &callback1, &ts);</font></font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"> <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">Task
- t4(1000, TASK_FOREVER, &callback4, &ts);</font></font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"> <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t3.delay(10);</font></font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Note that t1 and t3
- call the same callback method, and are shifted in time by 10 millis.
- So effectively callback1 will be called every 10 millis, but would be
- "sandwiched" between t2 and t4.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">3. Use short efficient
- callback methods written for cooperative multitasking.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">What that means is:</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">a)
- <b>DO NOT</b> use Arduino's <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt"><b>delay</b></font></font><font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">()</font></font><font face="Andale Mono, MS Mincho">
- </font>function. It is blocking and will hold the entire chain.
- Instead break the callback method into two, switch the callback
- method of the task where delay is necessary and delay the task by
- that number of millis. You get your delay, and other tasks get a
- chance to run:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">instead
- of:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- stuff</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">delay(1000);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- more stuff</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
-
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">do
- this:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback1() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- stuff</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.setCallback(&callback2);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.delay(1000);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback2() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- more stuff</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.setCallback(&callback1);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">b)
- Same goes to <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt"><b>pulseIn</b></font></font><font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">()</font></font>
- function. If you have to use it, set the timeout parameter such that
- it is not a default 1 second. PulseIn functionality could be achieved
- via pin interrupts, and that solution is non-blocking.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">c)
- Do don run long loops (for or do/while) in you callback methods. Make
- the main arduino loop be the loop driver for you:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">instead
- of:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
-
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">for(int
- i=0; i<1000; i++) {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- stuff // one loop action</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">do
- this:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">Task
- t1(TASK_IMMEDIATE, 1000, &callback);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
-
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">int
- i = t1.getRunCounter() -1;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- stuff // one loop action</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">or
- this:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">Task
- t1(TASK_IMMEDIATE, 1000, &callback, true, &t1On);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
-
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">int
- i;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">bool
- t1On() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">i
- = 0;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">return
- true;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
-
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- stuff // one loop action</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">i++;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" align="center" style="margin-left: 0.49in; margin-bottom: 0in">
- <b>REMEMBER: you are already inside the loop - take advantage of it. </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">d)
- Break long running callback methods into several shorter ones, and
- pass control from one to the other via <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt"><b>setCallback</b></font></font><font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">()
- </font></font>method:</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">Task
- t1(TASK_IMMEDIATE, TASK_FAREVER, &callback);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
-
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- do some stuff</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.setCallback(&callback_step2);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback_step2() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- do more stuff</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.setCallback(&callback_step3);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback_step3() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- do last part of the stuff</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.setCallback(&callback);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.delay(1000);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">This
- will execute all parts of the callback function in three successive
- steps, scheduled immediately, but allowing other tasks in the chain
- to run. Notice that task is scheduled to run immediately, and 1
- second period is achieved by delaying the task for 1000 millis at the
- last step.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Alternatively
- you could schedule the task to run every 1000 millis and use
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">forceNextIteration()</font></font>
- method in steps 1 and 2 (but not 3!)</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">Task
- t1(1000, TASK_FOREVER, &callback);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
-
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- do some stuff</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.setCallback(&callback_step2);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.forceNextIteration();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback_step2() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- do more stuff</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.setCallback(&callback_step3);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.forceNextIteration();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">void
- callback_step3() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">...
- do last part of the stuff</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">t1.setCallback(&callback);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">e)
- Compile the library with <font face="FreeMono, MS Mincho, monospace"><font size="2" style="font-size: 10pt"><b>_TASK_TIMECRITICAL</b></font></font>
- enabled and check if your tasks are falling behind schedule. If they
- are - you need to optimize your code further (or maybe re-evaluate
- your schedule). If they are not - all is well and you don't need to
- do anything. E.g., I have a spider robot which needs to measure
- distance, control motors, and keep track of the angle via querying
- gyroscope and accelerometer every 10 ms. The idea was to flash
- onboard LED if any of the tasks fall behind. At 10 ms interval for
- the gyro the LED does not flash, which means none of the tasks are
- blocking the others from starting on time.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <font size="4" style="font-size: 14pt"><b>API DOCUMENTATION:</b></font></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <font size="4" style="font-size: 14pt"><b>TASKS:</b></font></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in">CREATION:</p>
- <p class="western" style="margin-bottom: 0in"><b>Task();</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Default
- constructor.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Takes
- no parameters and creates a task that could be scheduled to run at
- every scheduling pass indefinitely, but does not have a callback
- method defined, so no code execution will actually take place.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">All
- tasks are created <b>disabled</b> by default.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>Task(unsigned long aInterval, long aIterations, void
- (*aCallback)(), Scheduler* aScheduler, bool aEnable, bool
- (*aOnEnable)(), void (*aOnDisable)())</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Constructor
- with parameters.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Creates
- a task that is scheduled to run every <aInterval> milliseconds,
- <aIterations> times, executing <aCallback> method on
- every pass.
- </p>
- <ol>
- <li><p class="western" style="margin-bottom: 0in">aInterval is in
- milliseconds (or microseconds) (<b>default = 0)</b></p>
- <li><p class="western" style="margin-bottom: 0in">aIteration in
- number of times, -1 for indefinite execution (<b>default = -1)</b><br><b>Note:
- </b>Tasks do not remember the number of iteration set initially.
- After the iterations are done, internal iteration counter is 0. If
- you need to perform another set of iterations, you need to set the
- number of iterations again. <br><b>Note: </b>Tasks which performed
- all their iterations remain active.
- </p>
- <li><p class="western" style="margin-bottom: 0in">aCallback is a
- pointer to a void callback method without parameters (<b>default =
- NULL)</b></p>
- <li><p class="western" style="margin-bottom: 0in">aScheduler –
- <b>optional</b> reference to existing scheduler. If supplied (not
- NULL) this task will be appended to the task chain of the current
- scheduler). (<b>default = NULL)</b></p>
- <li><p class="western" style="margin-bottom: 0in">aEnable –
- <b>optional</b>. Value of <b>true </b>will create task enabled.
- (<b>default = false)</b></p>
- <li><p class="western" style="margin-bottom: 0in">aOnEnable is a
- pointer to a bool callback method without parameters, invoked when
- task is enabled. If OnEnable method returns <b>true</b>, task is
- enabled. If <b>OnEnable</b> method return <b>false</b>, task remains
- disabled (<b>default = NULL)</b></p>
- <li><p class="western" style="margin-bottom: 0in">aOnDisable is a
- pointer to a void callback method without parameters, invoked when
- task is disabled (<b>default = NULL)</b></p>
- </ol>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">All
- tasks are created <b>disabled</b> by default (unless <b>aEnable</b> =
- true). You have to explicitly enable the task for execution.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- </b>OnEnable callback method is called immediately when task is
- enabled, which could be well ahead of the scheduled execution time of
- the task. Please bear that in mind – other tasks, hardware, serial
- interface may not even be initialized yet. It is always advisable to
- explicitly enable tasks with OnEnable methods after all
- initialization methods completed (e.g., at the end of <b>setup</b>()
- method)
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Enabled
- task is scheduled for execution as soon as the Scheduler's <b>execute</b>()
- methods gets control. In order to delay first run of the task, use
- <b>enableDelayed</b> or <b>delay</b> method (for enabled tasks)
- method.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>Task(void (*aCallback)(), Scheduler* aScheduler, bool
- (*aOnEnable)(), void (*aOnDisable)())</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- compiled with support for Status Request objects, this constructor
- creates a Task for activation on event (since such tasks must run
- <b>waitFor() </b>method, their <i>interval</i>, <i>iteration</i> and
- <i>enabled</i> status will be set by that method (<i>to 0, 1 and
- false</i> respectively).</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>INFORMATION</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">The
- following 3 “getter” methods return task status
- (enabled/disabled), execution interval in milliseconds, number of
- <i><b>remaining</b></i> iterations.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>bool
- isEnabled() </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>unsigned
- long getInterval()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>long
- getIterations() </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>long getStartDelay()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- library is compiled with <font face="Courier New, monospace">_TASK_TIMECRITICAL</font>
- enabled, you can assess how much later the callback method was
- invoked against when it was scheduled to be invoked. The return value
- of <b>getStartDelay () </b>method provides this information in
- milliseconds (or microseconds).
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>long getOverrun()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- library is compiled with <font face="Courier New, monospace">_TASK_TIMECRITICAL</font>
- enabled, tasks are monitored for “long running” scenario. A “long
- running” task is a task that does not finish processing its
- callback methods quickly, and thus creates a situation for itself and
- other tasks where they don't run on a scheduled interval, but rather
- “catch up” and are behind. When task scheduler sets the next
- execution target time, it adds Task's execution interval to the
- previously scheduled execution time:</p>
- <p class="western" align="center" style="margin-left: 0.49in; margin-bottom: 0in">
- <b>next execution time = current execution scheduled time + task
- execution interval</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- <b>next execution time</b> happens to be already in the past (<b>next
- execution time</b> < <b>millis()</b>), then task is considered
- <i><b>overrun</b></i>. <b>GetOverrun</b> method returns number of
- milliseconds between next execution time and current time. If the
- <b>value is negative</b>, the task has overrun (cut into the) next
- execution interval by that many milliseconds.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Positive
- value indicate number of milliseconds (or microseconds) of slack this
- task has for execution purposes.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>unsigned long
- getRunCounter()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Returns
- the number of the current run. “Current run” is the number of
- times a callback method has been invoked since the last time a task
- was enabled. <br><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- </b>The <b>runCounter</b> value is incremented <i>before</i> callback
- method is invoked. If a task is checking the <b>runCounter</b> value
- within its callback method, then the first run value is 1.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- task T1 is checking the <b>runCounter</b> value of another task (T2)
- , then value = 0 indicates that T2 has not been invoked yet, and
- value = 1 indicates that T2 has run once.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>bool
- isFirstIteration()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Indicates
- whether current pass is (or will be) a first iteration of the task.
- </p>
- <p class="western" style="margin-bottom: 0in"><b>bool
- isLastIteration()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">For
- tasks with a l<i>imited number of iterations only</i>, indicates
- whether current pass is the last iteration.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>CONTROL:</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void enable();</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Enables
- the task, and schedules it for immediate execution (without delay) at
- this or next scheduling pass depending on when the task was enabled.
- Scheduler will execute the next pass without any delay because there
- is a task which was enabled and requires execution.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- </b>if task being enabled is not assigned to a scheduler and is not
- part of execution chain, then task <b>will not</b> be enabled.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- </b>enable() invokes task’s <b>OnEnable</b> method (if not NULL)
- <b>immediately</b>, which can prepare task for execution. <b>OnEnable</b>
- must return a value of <b>true</b> for task to be enabled. If
- <b>OnEnable</b> returns <b>false</b>, task remains disabled.
- <b>OnEnable</b> is invoked every time <b>enable</b> is called,
- regardless if task is already enabled or not. Alignment to current
- millis() is performed after <b>OnEnable</b> exits, so any changes to
- the interval inside <b>OnEnable</b> is taken into consideration.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">TaskScheduler
- allows tasks to be added to the a Scheduler and enabled at the time
- of creation. <b>Be very careful</b> with such tasks – the <b>OnEnable
- </b>method will be executed immediately, while certain objects (i.e.,
- other Tasks, libraries) are not yet ready (e.g., <b>Wire.begin()</b>
- was not yet called), or hardware not yet activated (pins not set to
- INPUT or OUTPUT).
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">It
- is very much recommended to to enable all tasks at the end of <b>setup()</b>
- method after all initializations are done.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- you require immediate execution of already enabled task, use
- <b>forceNextIteratoin</b>() method instead of <b>enable</b>(): it
- achieves the result, but does not call <b>OnEnable</b> method.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:</b>
- in the event <b>enable</b>() method is called inside the <b>OnEnable</b>
- callback method (thus basically creating indefinte loop),
- TaskScheduler will only call <b>OnEnable</b> once (thus protecting
- the Task against <b>OnEnable</b> infinite loop).
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- </b>internal StatusRequest object will be set waiting for an event
- when Task is enabled (if TaskScheduler is compiled with support for
- StatusRequests). StatusRequest object is set waiting <b>after </b>the
- call to onEnable() method of the Task (if defined). Consequently, any
- Task#2 that is expected to wait on this Task’s internal
- StatusRequest should do it only <b>after </b>this task is enabled.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>bool enableIfNot();</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Enables
- the task only if it was previously disabled. Returns previous enable
- state: <b>true</b> if task was already enabled, and <b>false</b> if
- task was disabled. Since <b>enable() </b>schedules Task for execution
- immediately, this method provides a way to activate tasks and
- schedule them for immediate execution only if they are not active
- already.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">All
- <b>NOTES</b> from the enable() method apply.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void delay();</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Schedules
- the task for execution after a delay (aInterval), but does not change
- the enabled/disabled status of the task.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- </b>a delay of 0 (zero) will delay task for current execution
- interval. Use <b>forceNextIteration() </b>method to force execution
- of the task’s callback during immediate next scheduling pass.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void
- </b><b>forceNextIteration</b><b>();</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Schedules
- the task for execution during immediate next scheduling pass.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">The
- Task must be already <i>enabled</i> prior to this method.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>Note:
- </b>Task’s schedule is adjusted to run from this moment in time.
- For instance: if a task was running every 10 seconds: 10, 20, 30, ..,
- calling <b>forceNextIteration </b>at 44th second of task execution
- will make subsequent schedule look like: 44, 54, 64, 74, ..</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void
- enableDelayed();</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Enables
- the task, and schedules it for execution after task's current
- scheduling interval (aInterval).
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void enableDelayed
- (unsigned long aDelay);</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Enables
- the task, and schedules it for execution after a specific delay
- (aDelay, which may be different from aInterval).
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void restart();</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">For
- tasks with limited number of iterations only, <b>restart</b> method
- will re-enable the task, set the number of iterations back to last
- set value, and schedule task for execution as soon as possible.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void restartDelayed
- (unsigned long aDelay);</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Same
- as <b>restart()</b> method, with the only difference being that Task
- is scheduled to run first iteration after a delay = <b>aDelay</b>
- milliseconds (or microseconds).</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>bool disable();</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Disables
- the task. Scheduler will not execute this task any longer, even if it
- remains in the chain. Task <b>can</b> be later re-enabled for
- execution.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Return
- previous enabled state: <b>true</b> if task was enabled prior to
- calling disable, and <b>false</b> otherwise.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- not NULL, task’s <b>OnDisable</b> method is invoked <b>immediately</b>.
- <b>OnDisable</b> is invoked only if task was in enabled state.
- Calling <b>disable</b> 3 times for instance will invoke <b>OnDisable</b>
- only once.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- </b>internal StatusRequest object will signal completion of an event
- when Task is disabled (if TaskScheduler is compiled with support for
- StatusRequests). StatusRequest object is set complete <b>after </b>the
- call to onDisable() method of the Task (if defined). Consequently,
- the task which has to signal its completion to other Tasks could not
- restart itself. Do so will not ever set the internal StatusRequest
- object to a complete status, since the Task is never really disabled.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void set(unsigned
- long aInterval, long aIterations, void (*aCallback)() , bool
- (*aOnEnable)() , void (*aOnDisable)());</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Allows
- dynamic control of task execution parameters in one method call.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>Note</b><b>:
- </b>OnEnable and OnDisable parameters can be omitted. In that case
- they will be assigned to <b>NULL</b> and respective methods will no
- longer be called. Therefore it is advisable to use either all five
- parameters explicitly, or employ individual “setter” methods
- below instead.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- Next five “setter” methods allow changes of individual task
- execution control parameters.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>void
- setInterval (unsigned long aInterval) </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>void
- setIterations (long aIterations) </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>void
- setCallback (void (*aCallback)()) </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>void
- setOnEnable (bool (*aCallback)()) </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>void
- setOnDisable (void (*aCallback)()) </b>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>NOTE: </b>Next
- execution time calculation takes place <b>after</b> the callback
- method is called, so new interval will be used immediately by the
- scheduler. For the situations when one task is changing the interval
- parameter for the other, <b>setInterval</b> method calls <b>delay
- </b>explicitly to guarantee schedule change, however it <b>does not
- </b>enable the task if task is disabled.</p>
- <p class="western" style="margin-bottom: 0in"><b>NOTE: </b>Tasks that
- ran through all their allocated iterations are disabled.
- <b>SetIterations()</b> method <b>DOES NOT</b> enable the task. Either
- <b>enable</b> explicitly, or use <b>restart</b> methods.
- </p>
- <p class="western" style="margin-bottom: 0in">Please note that as a
- result execution of the taks is <b>delayed</b> by the provided
- interval. If immediate invocation is required, call
- <b>forceNextIteration</b>() method after setting a new interval.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void yield(void
- (*aCallback)())</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">This method could be
- used to break up long callback methods. A long callback method should
- be broken up into several shorter methods. Yield method just gives
- control back to scheduler, while ensuring that next iteration of the
- Task is executed immediately with the next callback method. Basically
- “yield(&callBack2)” is equivalent to setting new callback
- method, and forcing next iteration to be immediate. Please not that
- original interval and number of iterations are preserved. Even the
- runcounter of the callback2 after yielding will remain the same.
- Typically a call to yield() method is the last line of the method
- yielding.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void yield</b><font face="Times New Roman, serif"><b>Once</b></font><b>(void
- (*aCallback)())</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">This method is
- equivalent to <b>yield(),</b> only execution of the target <i><b>a</b></i><font face="Times New Roman, serif"><span lang="ru-RU"><i><b>С</b></i></span></font><i><b>allback</b></i>
- method is set to happen only once, after which the Task will be
- disabled.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>STATUS REQUEST METHODS:</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>void waitFor(StatusRequest* aStatusRequest, unsigned long
- aInterval = 0, long aIterations = 1)</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>void waitForDelayed(StatusRequest* aStatusRequest, unsigned long
- aInterval = 0, long aIterations = 1)</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- compiled with support for Status Requests, these methods make task
- wait for the completion of <b>aStatusRequest</b> event.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">By
- default <b>waitFor() </b>sets tasks interval to <b>0 (zero)</b> for
- immediate execution when event happens, and also sets the number of
- <b>iterations to 1</b>. However, you can specify different interval
- and number of iterations.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">By
- default <b>waitForDelayed() </b>sets tasks interval to a supplied
- value or (if omitted or zero) keeps the current interval, so delayed
- execution will take place when the event happens. It also sets the
- number of <b>iterations to 1</b> by default if not supplied.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">When
- Status Request object completes, all tasks waiting on it are executed
- during next scheduling pass. Tasks waiting via <b>waitFor</b>()
- method are executed immediately. Tasks waiting via <b>waitForDelayed</b>()
- method are activated, but executed after current or supplied interval
- delay.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>Note:
- aStatusRequest</b> should be “activated” by calling <b>setWaiting()
- </b>method before making a task wait on it. Otherwise, the task will
- execute immediately.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">The
- sequence of events to use Status Request object is as follows:</p>
- <ol>
- <li><p class="western" style="margin-bottom: 0in">Create a status
- request object</p>
- <li><p class="western" style="margin-bottom: 0in">Activate status
- request object (calling its <b>setWaiting</b>() method)</p>
- <li><p class="western" style="margin-bottom: 0in">Set up tasks to
- wait of the event completion</p>
- <li><p class="western" style="margin-bottom: 0in">Signal completion
- of event(s)</p>
- </ol>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>StatusRequest*
- getStatusRequest()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Returns
- a pointer to StatusReqeust object this Task was waiting on.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>StatusRequest*
- getInternalStatusRequest()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Returns
- a pointer to an internal StatusReqeust object associated with this
- Task. Internal StatusRequest object is:</p>
- <ol>
- <li><p class="western" style="margin-bottom: 0in">Always waits on 1
- event – completion of this task</p>
- <li><p class="western" style="margin-bottom: 0in">Is activated (set
- to “waiting” status) after Task is enabled</p>
- <li><p class="western" style="margin-bottom: 0in">Is completed after
- Task is disabled (either explicitly, or by running out of
- iterations)</p>
- </ol>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- </b>Please remember that a task is deactivated at the next scheduling
- pass after the last iteration, which means that other Tasks in the
- chain will have a chance to run before Task StatusRequest signaling
- completion of the internal StatusRequest. However, there is no
- further delay – deactivation will take place at the next scheduling
- pass.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>TASK ID, CONTROL POINTS METHODS:</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>void setId(unsigned int aID);</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- compiled with support for Task IDs, this method will set the task ID
- explicitly.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Calling
- this method is not necessary as task IDs are assigned automatically
- during task creation: 1, 2, 3, …</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>unsigned int getId();</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- compiled with support for Task IDs, this method return current task’s
- ID.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>void setControlPoint (unsigned int aPoint);</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- compiled with support for Task IDs, this method will set a control
- point in the task’s code. Control points are similar to “try…catch”
- blocks, with control point ID specifying where in the code the “try”
- part started, and a mechanism like watchdog timer providing the
- “catch” functionality.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>unsigned int getControlPoint()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in; page-break-after: avoid">
- If compiled with support for Task IDs, this method will return
- currently set control point for this task.</p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>LOCAL TASK STORAGE
- METHODS:</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>void setLtsPointer(void *aPtr);</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- compiled with support for LTS, this method will set the task's local
- storage pointer.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>void *getLtsPointer();</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- compiled with support for LTS, this method will return reference to
- the task's local storage.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>Note:
- </b>the value returned has type (void *), and needs to be re-cast
- into appropriate pointer type. Please refer to example sketches for
- implementation options.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>STATUS REQUEST:</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>CREATION:</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>StatusRequest()</b><br><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Default
- constructor.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Takes
- no parameters. Creates Status Request object, which is assigned a
- status of “completed” on creation.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void
- setWaiting(unsigned int aCount = 1)</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Activates
- Status Request object. By default each object is set to wait on one
- event only, however, if <b>aCount</b> is supplied, Status Request can
- wait on multiple events. For instance, <b>setWaiting(3)</b> will wait
- on three signals. An example could be waiting for completion of
- measurements from 3 sensors.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>bool signal(int
- aStatus)</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Signals
- completion of the event to the Status Request object, and passes a
- completion code, which could be interrogated later.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>Note:
- </b> passing a <b>negative</b> status code to the status request
- object is considered reporting an error condition, and will complete
- the status request regardless of how many outstanding signals it is
- still waiting for.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>Note</b>:
- only the latest status code is kept.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>bool signalComplete
- (int aStatus)</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Signals
- completion of <b>ALL</b> events to the Status Request object, and
- passes a completion code, which could be interrogated later. The
- status request completes regardless of how many events it is still
- waiting on.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>bool pending() </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Returns
- <b>true</b> if status request is still waiting for event or events to
- happen.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>bool completed () </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Returns
- <b>true</b> if status request event has completed.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>int getStatus()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Returns
- the status code passed to the status request object by the <b>signal()
- </b>and <b>signalComplete() </b>methods.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Any
- <b>positive</b> number is considered a successful completion status.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">A
- 0 (zero) is considered a default successful completion status.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Any
- <b>negative</b> number is considered an error code and unsuccessful
- completion of a request.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>int getCount()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Returns
- the count of events not yet completed. Typically by default a
- StatusRequest object only waits on 1 event. However, in the
- situations where a StatusRequest object is waiting on multiple
- events, a number of events <b>not yet completed </b>is returned by
- this method.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-inside: avoid; page-break-after: avoid">
- <font size="4" style="font-size: 14pt"><b>TASK SCHEDULER:</b></font></p>
- <p class="western" style="margin-bottom: 0in; page-break-inside: avoid; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-inside: avoid; page-break-after: avoid">
- <b>CREATION:</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-inside: avoid; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>Scheduler()</b><br><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Default
- constructor.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Takes
- no parameters. Creates task scheduler with default parameters and an
- empty task queue.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void init()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Initializes
- the task queue and scheduler parameters, Executed as part of
- constructor, so don't need to be explicitly called after creation.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>Note:
- </b>be default (if compiled with <font face="Courier New, monospace">_TASK_TIMECRITICAL</font>
- enabled) scheduler is allowed to put processor to IDLE sleep mode. If
- this behavior was changed via <b>allowSleep()</b> method, <b>inti()
- </b>will <b>NOT</b> reset allow sleep particular parameter.
- </p>
- <p class="western" style="margin-bottom: 0in"><br><b>void
- addTask(Task& aTask)</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Adds
- task aTask to the execution queue (or chain) of tasks by appending it
- to the end of the chain. If two tasks are scheduled for execution,
- the sequence will match the order in which tasks were appended to the
- chain. However, in reality, due to different timing of task
- execution, the actual order may be different.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>Note:
- </b>Currently, changing the execution sequence in a chain dynamically
- is not supported.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- you need to reorder the chain sequence – initialize the scheduler
- and re-add the tasks in a different order.
- </p>
- <p class="western" style="margin-bottom: 0in"><br><b>void
- deleteTask(Task& aTask)</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Deletes
- task aTask from the execution chain. The chain of remaining tasks is
- linked together (i.e</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">if
- original task chain is 1 → 2 → 3 → 4, deleting 3 will result in
- 1 → 2 → 4).</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>Note:
- </b>it is not required to delete a task from the chain. A disabled
- task will not be executed anyway, but you save a few microseconds per
- scheduling pass by deleting it, since it is not even considered for
- execution.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">An
- example of proper use of this method would be running some sort of
- <b>initialize</b> task in the chain, and then deleting it from the
- chain since it only needs to run once.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void allowSleep(bool
- aState) </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Available
- in API only if compiled with <font face="Courier New, monospace">_TASK_TIMECRITICAL</font>
- enabled. Controls whether scheduler is allowed (<b>aState =true</b>),
- or not (<b>aState =false</b>) to put processor into IDLE sleep mode
- in case not tasks are scheduled to run.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">The
- <b>default</b> behavior of scheduler upon creation is to allow sleep
- mode.
- </p>
- <p class="western" style="margin-bottom: 0in"><br><b>void
- enableAll(bool aRecursive= true)</b></p>
- <p class="western" style="margin-bottom: 0in"><b>void disableAll(bool
- aRecursive= true)</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">enables
- and disables (respectively) all tasks in the chain. Convenient if
- your need to enable/disable majority of the tasks (i.e. disable all
- and then enable one).
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- support for layered task priority is enabled, supplying <b>aRecursive</b>
- parameter will enable/disable higher priority tasks as well (<b>true</b>,
- default), or tasks only on this priority layer (<b>false</b>).
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br><b>Task& currentTask()<br></b><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Returns
- reference to the task, currently executing via <b>execute()</b> loop
- <b>OR </b>for OnEnable and OnDisable methods, reference to the task
- being enabled or disabled.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">This
- distinction is important because one task can activate the other, and
- OnEnable should be referring to the task being enabled, not being
- executed.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Could
- be used by callback methods to identify which Task actually invoked
- this callback method.</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>void* currentLts()<br></b><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Returns
- pointer to Local Task Storage of the task, currently executing via
- <b>execute()</b> loop <b>OR </b>for OnEnable and OnDisable methods,
- task being enabled or disabled.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void startNow (bool
- aRecursive= true)</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Sets
- <b>ALL active tasks</b> in the execution chain to start execution
- immediately. Should be placed at the end of <b>setup()</b> method to
- prevent task execution race due to long running setup tasks (hardware
- initialization, etc.) following task activation.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- support for layered task priority is enabled, supplying <b>aRecursive</b>
- parameter will set immediate execution for higher priority tasks as
- well (<b>true</b>, default), or tasks only on this priority layer
- (<b>false</b>).
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- </b>if <b>setup()</b> method does not contain long running tasks, use
- of <b>startNow()</b> method is not necessary. Alternatively, all
- tasks could be enabled <b>after </b>long-running <b>setup()</b>
- processes, thus eliminating the need to use <b>startNow()</b> method.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- </b>Any tasks which should execute after a <i><b>delay</b></i>,
- should be explicitly delayed after call to <b>startNow()</b> method.
- </p>
- <p class="western" style="margin-bottom: 0in"><br><b>bool execute()</b></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Executes
- one scheduling pass, including (in case of the base priority
- scheduler) end-of-pass sleep. This method should be placed inside the
- <b>loop()</b> method of the sketch. Since <b>execute</b> exits after
- every pass, you can put additional statements after <b>execute</b>
- inside the <b>loop().</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>If
- layered task prioritization is enabled, all higher priority tasks
- will be evaluated and invoked by the base </b><font face="Courier New, monospace"><b>execute()</b></font><b>
- method. There is no need to call </b><font face="Courier New, monospace"><b>execute()</b></font><b>
- of the higher priority schedulers explicitly. </b>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">Generally,
- base priority execute will perform the following steps:</p>
- <ol>
- <li><p class="western" style="margin-bottom: 0in">Call higher
- priority scheduler’s execute method, if provided.</p>
- <li><p class="western" style="margin-bottom: 0in">Ignore task
- completely if it is disabled.</p>
- <li><p class="western" style="margin-bottom: 0in">Disable task if it
- ran out of iterations (calling OnDisable, if necessary).</p>
- <li><p class="western" style="margin-bottom: 0in">Check if task is
- waiting on a StatusRequest object, and make appropriate scheduling
- arrangements</p>
- <li><p class="western" style="margin-bottom: 0in">Perform necessary
- timing calculations</p>
- <li><p class="western" style="margin-bottom: 0in">Invoke task's
- callback method, if it is time to do so, and one is provided.
- </p>
- <li><p class="western" style="margin-bottom: 0in">Put
- microcontroller to sleep (if requested and supported) if none of the
- tasks were invoked.
- </p>
- </ol>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>Please
- NOTE:</b> schedule-related calculations are performed prior to task's
- callback method invocation. This allows tasks to manipulate their
- runtime parameters (like execution interval) directly.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>bool isOverrun()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- library is compiled with <font face="Courier New, monospace">_TASK_TIMECRITICAL</font>
- enabled, this method returns <b>true</b> if currently invoked task
- has overrun its scheduled start time when it was invoked. Returns
- <b>false</b> if task has been invoked according to schedule.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>TASK PRIORITY:</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>void
- setHighPriorityScheduler(Scheduler* aScheduler);</b><br><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- library is compiled with <font face="Courier New, monospace">_TASK_PRIORITY
- </font>enabled, this method associates current scheduler with a
- higher priority scheduler.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- </b>Only one <b>execute() </b>method needs to be explicitly called in
- the main <b>loop(). </b>That is the execute method of <b>base</b>
- priority scheduler. All higher priority schedulers are called by the
- base priority scheduler.</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><b>static Scheduler&
- currentScheduler()</b></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">If
- library is compiled with <font face="Courier New, monospace">_TASK_PRIORITY
- </font>enabled, this method returns reference to a scheduler, which
- invoked current task.
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><b>NOTE:
- Please refer to examples 11 and 12 for illustration of Task Priority
- functionality</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>CONSTANTS:</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in; page-break-after: avoid">
- <b>TASK_SECOND (1000 millis or 1000000 micros)</b></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">Task
- interval of 1 second</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in; page-break-after: avoid">
- <b>TASK_MINUTE (60000 millis or 60000000 micros)</b></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">Task
- interval of 1 minute</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in; page-break-after: avoid">
- <b>TASK_HOUR (3600000 millis or 3600000000 micros)</b></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">Task
- interval of 1 hour</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in; page-break-after: avoid">
- <b>TASK_FOREVER (-1)</b></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">Task
- number of iterations for infinite number of iterations</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in; page-break-after: avoid">
- <b>TASK_ONCE (1)</b></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">Task
- single iteration</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in; page-break-after: avoid">
- <b>TASK_IMMEDIATE (0)</b></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">Task
- interval for immediate execution</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in; page-break-before: always; page-break-after: avoid">
- <b>IMPLEMENTATION SCENARIOS AND IDEAS:</b></p>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <ol>
- <li><p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <b>EVENT DRIVEN PROGRAMMING</b></p>
- </ol>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in">Each of the processes
- of your application becomes a separate and distinct programming area,
- which may or may not interact and control each other.
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Example:
- </p>
- <p class="western" style="margin-bottom: 0in">In a plant watering
- system you need to measure soil humidity, control pump and display
- the results</p>
- <p class="western" style="margin-bottom: 0in">Each of the areas
- becomes a task:</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- </font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><b>tMeasure</b></font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">
- (TMEASURE_INTERVAL*SECOND, TASK_FOREVER, &measureCallback);<br>Task
- </font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><b>tWater</b></font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">
- (TWATER_INTERVAL*SECOND, RETRIES, &waterCallback);<br>Task
- </font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><b>tDisplay</b></font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">
- (TDISPLAY_INTERVAL*SECOND, TASK_FOREVER, &displayCallback); <br></font></font><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Scheduler
- </font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><b>taskManager</b></font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">;</font></font><br><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Further, once you turn
- on the pump, you keep it running for TWATER_INTERVAL interval and
- then turn it off. Turning off a pump is also a task which only needs
- to run once for every time the pump is turned on:</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- </font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><b>tWaterOff</b></font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">
- (WATERTIME*SECOND, TASK_ONCE,</font></font><font face="Courier New, monospace">
- &waterOffCallback);</font><br><br>
- </p>
- <p class="western" style="margin-bottom: 0in">Example of the callback
- method:</p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- waterOffCallback() {<br> motorOff();<br>
- tWater.enableDelayed();<br>}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">or</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- waterCallback() {<br> if (tWater.getIterations()) {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- If this is not the last iteration = turn the pump on<br>
- motorOn();<br> tWaterOff.set(parameters.watertime * TASK_SECOND,
- TASK_ONCE, &waterOffCallback);<br>
- tWaterOff.enableDelayed();<br> return;<br> }</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- We could not reach target humidity – something is wrong<br>
- motorOff;<br> taskManager.disableAll();<br> tError.enable();<br>}</font></font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">Your
- sample </font><font face="Times New Roman, serif"><b>setup</b></font><font face="Times New Roman, serif">()
- and </font><font face="Times New Roman, serif"><b>loop</b></font><font face="Times New Roman, serif">()
- (partially) are as follows. </font>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif"><b>Note:
- </b></font><font face="Times New Roman, serif">please note that
- tWater is </font><font face="Times New Roman, serif"><b>not</b></font><font face="Times New Roman, serif">
- activated during setup(). It is activated by tMeasure callback once
- the watering conditions are met. </font>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 10pt"> void
- </font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">setup()</font></font></p>
- <p class="western" style="margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">
- ...</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tWater.setIterations(parameters.retries);<br>
- tWaterOff.setInterval(parameters.watertime * SECOND);<br><br><br>
- taskManager.init();<br> taskManager.addTask(tMeasure);<br>
- taskManager.addTask(tDisplay);<br> taskManager.addTask(tWater);<br>
- taskManager.addTask(tWaterOff);<br> <br> tMeasure.enable();<br>
- tDisplay.enable();<br><br> currentHumidity =
- measureHumidity();<br>}<br><br><br>void loop ()<br>{<br>
- taskManager.execute();<br>}</font></font><font face="Times New Roman, serif"><font size="2" style="font-size: 10pt"><br></font></font><br>
- </p>
- <ol start="2">
- <li><p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- “<font face="Times New Roman, serif"><b>NATIVE” SUPPORT FOR
- FINITE STATE MACHINE</b></font></p>
- </ol>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">Define
- “states” as callback method or methods. Each callback method
- executes activities specific to a “state” and then “transitions”
- to the next state by assigning next callback method to the task. </font>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">Transition
- from one state to the next is achieved by setting next callback
- method at the end of preceding one. </font>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif"><b>Note:
- </b></font><font face="Times New Roman, serif">do not call the next
- callback method explicitly. Yield to the scheduler, and let the
- scheduler take care of next iteration during the next pass. (Thus
- giving other tasks change to run their callback methods). </font>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">Example:
- Blinking LED 2 times a second could be achieved this way</font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Scheduler
- ts;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- tLedBlinker (500, TASK_FOREVER, &ledOnCallback, &ts, true);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- ledOnCallback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> turnLedOn();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> tLedBlinker.setCallback(&ledOffCallback);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- ledOffCallback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> turnLedOff();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> tLedBlinker.setCallback(&ledOnCallback);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">setup()
- {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">loop
- () {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> ts.execute();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font><font face="Times New Roman, serif"><br></font><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">Obviously
- the example is simple, but gives the idea of how the tasks could be
- used to go through states.</font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <ol start="3">
- <li><p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <font face="Times New Roman, serif"><b>MULTIPLE POSSIBLE CALLBACKS
- FOR TASK</b></font></p>
- </ol>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">There
- may be a need to select an option for callback method based on
- certain criteria, or randomly. </font>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">You
- can achieve that by defining an array of callback method pointers and
- selecting one based on the criteria you need. </font>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">Example:
- when a robot detects an obstacle, it may go left, right backwards,
- etc. Each of the “directions” or “behaviors” are represented
- by a different callback methods. </font>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">Another
- example of using multiple callbacks:</font></p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">You
- may need to “initialize” variables for a particular task.</font></p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">In
- this case, define a tasks with two callbacks:</font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- </font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><b>tWork</b></font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">
- (T_INTERVAL, TASK_FOREVER, &workCallbackInit);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">…</p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- workCallbackInit() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> //
- do your initializationstuff here</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> </font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> //
- finally assigne the main callback method </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> tWork.setCallback(&workCallback);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- workCallback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> //
- main callback method</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> …</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">The
- task will initialize during first execution pass and switch to
- “regular” callback execution starting with second pass. There is
- a delay between first and second passes of the task (scheduling
- period, if defined). In order to execute the second pass immediately
- after initialization first pass, change the above code like this:</font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- workCallbackInit() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> //
- do your initializationstuff here</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> </font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> //
- finally assigne the main callback method </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> tWork.setCallback(&workCallback);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> tWork.enable();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">The
- task will run initialization first, then immediately second pass, and
- then switch to processing at regular intervals starting with a third
- pass. </font>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <ol start="3">
- <li><p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <font face="Times New Roman, serif"><b>INTERRUP-DRIVEN EXECUTION
- SUPPORT </b></font>
- </p>
- </ol>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">In
- case of interrupt-driven program flow, tasks could be scheduled to
- run once to request asynchronous execution (request), and then
- re-enabled (restarted) again with a different callback method to
- process the results. </font>
- </p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif"><b>Example</b></font><font face="Times New Roman, serif">:
- event driven distance calculation for ultrasonic pulses. EchoPin #6
- triggers pin change interrupts on rising and falling edges to
- determine the length of ultrasonic pulse.</font></p>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">#include
- <DirectIO.h><br>#include <TaskScheduler.h><br>#include
- <PinChangeInt.h><br><br><br>#define TRIGGERPIN 5<br>#define
- ECHOPIN 6<br><br>Output<TRIGGERPIN>
- pTrigger;<br>Input<ECHOPIN> pEcho;<br><br>Scheduler
- r;<br><br>Task tMeasure(TASK_SECOND, TASK_FOREVER, &measureCallback,
- &r, true);<br>Task tDisplay(TASK_SECOND, TASK_FOREVER,
- &displayCallback, &r, true);<br>Task tPing(TASK_IMMEDIATE,
- TASK_ONCE, &pingCalcCallback, &r, false);<br><br><br>volatile
- bool pulseBusy = false;<br>volatile bool pulseTimeout =
- false;<br>volatile unsigned long pulseStart = 0;<br>volatile unsigned
- long pulseStop = 0;<br>volatile unsigned long pingDistance =
- 0;<br><br><br>void pingTrigger(unsigned long aTimeout) {<br> if
- (pulseBusy) return; // do not trigger if in the middle of a pulse<br>
- if (pEcho == HIGH) return; // do not trigger if ECHO pin is high<br>
- <br> pulseBusy = true;<br> pulseTimeout = false;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><br>
- pTrigger = LOW;<br> delayMicroseconds(4);<br> pTrigger = HIGH;<br><br>
- tPing.setInterval (aTimeout);<br><br> delayMicroseconds(10);<br>
- pTrigger = LOW; <br></font></font><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tPing.restartDelayed();
- // timeout countdown starts now<br></font></font><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- will start the pulse clock on the rising edge of ECHO pin</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">PCintPort::attachInterrupt(ECHOPIN,
- &pingStartClock, RISING); <br>}<br><br></font></font><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- Start clock on the </font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><b>rising</b></font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">
- edge of the ultrasonic pulse<br>void pingStartClock() {<br>
- pulseStart = micros();<br> PCintPort::detachInterrupt(ECHOPIN); //
- not sure this is necessary<br> PCintPort::attachInterrupt(ECHOPIN,
- &pingStopClock, FALLING); <br> tPing.restartDelayed();<br>}<br><br>//
- Stop clock on the </font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><b>falling</b></font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">
- edge of the ultrasonic pulse<br>void pingStopClock() {<br> pulseStop
- = micros();<br> PcintPort::detachInterrupt(ECHOPIN);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">pingDistance
- = pulseStop - pulseStart;<br> pulseBusy = false;<br>
- tPing.disable(); // disable timeout<br>}<br></font></font><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- Stop clock because of the timeout – the wave did not return<br>void
- pingCalcCallback() {<br> if (pulseBusy) {<br> pingStopClock();<br>
- }<br> pulseTimeout = true;<br>}<br><br><br><br>// Initial measure
- callback sets the trigger<br>void measureCallback() {<br> if
- (pulseBusy) { // already measuring, try again<br>
- tMeasure.enable();<br> return;<br> }<br> pingTrigger(30); // 30
- milliseconds or max range of ~5.1 meters<br>
- tMeasure.setCallback(&measureCallbackWait);<br>}<br></font></font><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- Wait for the measurement to <br>void measureCallbackWait() {<br> if
- (pulseBusy) return;<br> tMeasure.setCallback(&measureCallback);
- <br>}<br><br><br>bool state = true;<br><br>void displayCallback() {<br>
- char d[256];<br> <br> unsigned long cm = pingDistance * 17 / 100;
- // cm<br> <br> snprintf(d, 256, "pulseStart =
- %8lu\tpulseStop=%8lu\tdistance, cm=%8lu", pulseStart, pulseStop,
- cm);<br> Serial.println(d);<br> <br>}<br><br>void setup() {<br> //
- put your setup code here, to run once:<br> <br>
- Serial.begin(115200);<br> <br><br> pTrigger = LOW;<br> pEcho =
- LOW;<br> <br>}<br><br>void loop() {<br> // put your main code here,
- to run repeatedly:<br> r.execute();<br>}<br></font></font><br>
- </p>
- <ol start="3">
- <li><p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <font face="Times New Roman, serif"><b>USING ONENABLE AND ONDISBALE
- METHODS </b></font>
- </p>
- </ol>
- <p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif">Consider
- a task to flash onboard LED for 5 seconds with random frequency. Task
- should be repeated every 30 seconds indefinitely. Since frequency is
- random, there are two challenges:</font></p>
- <ol start="3">
- <ol>
- <li><p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">We
- need to make sure LED is turned OFF at the last iteration</font></p>
- <li><p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif">We
- need to calculate random frequency every time</font></p>
- </ol>
- </ol>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif">Below
- is the implementation using TaskScheduler </font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><br></font></font><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">#define
- _TASK_SLEEP_ON_IDLE_RUN</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">#include
- <TaskScheduler.h></font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">#define
- LEDPIN 13</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Scheduler
- ts;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- tWrapper(30000, TASK_FOREVER, &WrapperCallback, &ts, true);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- tBlink(5000, TASK_ONCE, NULL, &ts, false, &BlinkOnEnable,
- &BlinkOnDisable);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- tLED(TASK_IMMEDIATE, TASK_FOREVER, NULL, &ts, false, NULL,
- &LEDOff);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- WrapperCallback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("In
- WrapperCallback");</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tBlink.restartDelayed();
- // LED blinking is initiated </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//every
- 30 seconds for 5 seconds</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- Upon being enabled, tBlink will define the parameters</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- and enable LED blinking task, which actually controls</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- the hardware (LED in this example)</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">bool
- BlinkOnEnable() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("In
- BlinkOnEnable");</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tLED.setInterval(
- 500 + random(501) );</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tLED.setCallback(
- &LEDOn);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tLED.enable();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">return
- true; // Task should be enabled</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- tBlink does not really need a callback method</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- since it just waits for 5 seconds for the first </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- and only iteration to occur. Once the iteration</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- takes place, tBlink is disabled by the Scheduler, </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- thus executing its OnDisable method below.</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- BlinkOnDisable() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("In
- BlinkOnDisable");</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tLED.disable();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- LEDOn () {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("In
- LEDOn");</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">digitalWrite(LEDPIN,
- HIGH);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tLED.setCallback(
- &LEDOff);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- LEDOff () {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("In
- LEDOff");</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">digitalWrite(LEDPIN,
- LOW);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tLED.setCallback(
- &LEDOn);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- Note that LEDOff method serves as OnDisable method</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- to make sure the LED is turned off when the tBlink</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- task finishes (or disabled ahead of time)</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- setup() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.begin(115200);
- </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">pinMode(LEDPIN,
- OUTPUT); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- loop() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- put your main code here, to run repeatedly:</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">ts.execute();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <ol start="3">
- <li><p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <font face="Times New Roman, serif"><b>USING STATUS REQUEST OBJECTS
- </b></font>
- </p>
- </ol>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">This
- test emulates querying 3 sensors once every 10 seconds, each could
- respond with a different delay (ultrasonic sensors for instance) and
- printing a min value of the three when all three have reported their
- values.</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">The
- overall timeout of 1 second is setup as well.</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">An
- error message needs to be printed if a timeout occurred instead of a
- value.</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">#define
- _TASK_SLEEP_ON_IDLE_RUN</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">#define
- _TASK_STATUS_REQUEST</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">#include
- <TaskScheduler.h></font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">StatusRequest
- measure;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Scheduler
- ts; </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- tCycle(10000, TASK_FOREVER, &CycleCallback, &ts, true);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- tMeasure(TASK_SECOND, TASK_ONCE, &MeasureCallback, &ts,
- false, &MeasureEnable, &MeasureDisable);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- tCalculate(&CalcCallback, &ts);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- tSensor1(TASK_IMMEDIATE, TASK_ONCE, &S1Callback, &ts, false,
- &S1Enable);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- tSensor2(TASK_IMMEDIATE, TASK_ONCE, &S2Callback, &ts, false,
- &S2Enable);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- tSensor3(TASK_IMMEDIATE, TASK_ONCE, &S3Callback, &ts, false,
- &S3Enable);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">long
- distance, d1, d2, d3;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- CycleCallback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("CycleCallback:
- Initiating measurement cycle every 10 seconds");</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tMeasure.restartDelayed();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">bool
- MeasureEnable() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("MeasureEnable:
- Activating sensors"); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">distance
- = 0;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">measure.setWaiting(3);
- // Set the StatusRequest to wait for 3 signals. </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tCalculate.waitFor(&measure);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tSensor1.restart();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tSensor2.restart();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tSensor3.restart();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">return
- true;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- MeasureCallback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("MeasureCallback:
- Invoked by calculate task or one second later"); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">if
- (measure.pending()) {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tCalculate.disable();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">measure.signalComplete(-1);
- // signal error</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("MeasureCallback:
- Timeout!");</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">else
- {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.print("MeasureCallback:
- Min distance=");Serial.println(distance);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- MeasureDisable() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("MeasureDisable:
- Cleaning up"); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tSensor1.disable();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tSensor2.disable();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tSensor3.disable();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- CalcCallback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("CalcCallback:
- calculating"); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">distance
- = -1;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">if
- ( measure.getStatus() >= 0) { // only calculate if statusrequest
- ended successfully</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">distance
- = d1 < d2 ? d1 : d2;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">distance
- = d3 < distance ? d3 : distance;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tMeasure.forceNextIteration();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">/**
- Simulation code for sensor 1</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"> <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">*
- ----------------------------</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">*/</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">bool
- S1Enable() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.print("S1Enable:
- Triggering sensor1. Delay="); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tSensor1.setInterval(
- random(1200) ); // Simulating sensor delay, which could go over 1
- second and cause timeout</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">d1
- = 0;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println(
- tSensor1.getInterval() );</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">return
- true;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- S1Callback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.print("S1Callback:
- Emulating measurement. d1="); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">d1
- = random(501); // pick a value from 0 to 500 "centimeters"
- simulating a measurement </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">measure.signal();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println(d1);
- </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">/**
- Simulation code for sensor 2</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"> <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">*
- ----------------------------</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">*/</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">bool
- S2Enable() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.print("S2Enable:
- Triggering sensor2. Delay="); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tSensor2.setInterval(
- random(1200) ); // Simulating sensor delay, which could go over 1
- second and cause timeout</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">d2
- = 0;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println(
- tSensor2.getInterval() );</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">return
- true;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- S2Callback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.print("S2Callback:
- Emulating measurement. d2="); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">d2
- = random(501); // pick a value from 0 to 500 "centimeters"
- simulating a measurement</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">measure.signal();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println(d2);
- </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">/**
- Simulation code for sensor 3</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"> <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">*
- ----------------------------</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">*/</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">bool
- S3Enable() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.print("S3Enable:
- Triggering sensor3. Delay="); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">tSensor3.setInterval(
- random(1200) ); // Simulating sensor delay, which could go over 1
- second and cause timeout</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">d3
- = 0;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println(
- tSensor3.getInterval() );</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">return
- true;</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- S3Callback() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.print("S3Callback:
- Emulating measurement. d3="); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">d3
- = random(501); // pick a value from 0 to 500 "centimeters"
- simulating a measurement</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">measure.signal();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println(d3);
- </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">/**
- Main Arduino code</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"> <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">*
- Not much is left here - everything is taken care of by the framework</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">*/</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- setup() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.begin(115200);</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Serial.println("TaskScheduler
- StatusRequest Sensor Emulation Test. Complex Test."); </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">randomSeed(analogRead(A1)+millis());</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- loop() {</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">ts.execute();</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <ol start="3">
- <li><p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <font face="Times New Roman, serif"><b>USING LOCAL TASK STORAGE
- POINTER </b></font>
- </p>
- </ol>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">Tasks
- can store a pointer to specific variable, structure or array, which
- represents variables specific for a particular task. This may be
- needed if you plan to use same callback method for multiple tasks.</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">Consider
- a scenario where you have several sensors of the same type. The
- actual process of triggering measurement and collecting information
- is identical. The only difference is the sensor address and a
- variable for storing the results. </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">In
- this case each of the tasks, which performs measurement will utilize
- the same callback methods. The only difference will be the variables
- (specific for each of the sensor). </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">Let's
- define a sensor data structure and declare a couple of variables (for
- 2 sensors for instance)</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p lang="en-US" class="western" style="margin-left: 0.98in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">typedef
- struct {</font></font></p>
- <p lang="en-US" class="western" style="margin-left: 0.98in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> unsigned
- int address;</font></font></p>
- <p lang="en-US" class="western" style="margin-left: 0.98in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> unsigned
- long distance;</font></font></p>
- <p lang="en-US" class="western" style="margin-left: 0.98in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}
- sensor_data;</font></font></p>
- <p lang="en-US" class="western" style="margin-left: 0.98in; margin-bottom: 0in">
- <br>
- </p>
- <p lang="en-US" class="western" style="margin-left: 0.98in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">sensor_data
- s1, s2; </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">Two
- separate tasks are running to collect sensor data.</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">(Note
- that both tasks refer to the same callback methods)</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p lang="en-US" class="western" style="margin-left: 0.98in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Scheduler
- ts;</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><span lang="en-US">Task
- t1(100, </span></font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">TASK_FOREVER</font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><span lang="en-US">,
- &Measure, &ts, false, &MeasureOn); </span></font></font>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><span lang="en-US">Task
- t2(100, </span></font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">TASK_FOREVER</font></font><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"><span lang="en-US">,
- &Measure, &ts, false, &MeasureOn); </span></font></font>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">Assign
- pointers to the respective variables in the </font></font><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt"><b>setup</b></font></font><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">()
- method:</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- setup() {</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 1.48in; margin-bottom: 0in">…</p>
- <p class="western" style="margin-left: 1.48in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">t1.setLtsPointer(&s1);</font></font></p>
- <p class="western" style="margin-left: 1.48in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">t2.setLtsPointer(&s2);</font></font></p>
- <p class="western" style="margin-left: 1.48in; margin-bottom: 0in">…</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">Obtain
- reference to specific </font></font><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt"><b>sensor_data</b></font></font><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">
- structure inside the common callback method:</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- Measure() {</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> Task&
- T = ts.currentTask(); </font></font>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> Sensor_data&
- V = *((sensor_data*) T.getLtsPointer());</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- For t1, V will be pointing at s1</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- For t2, V will be pointing at s2</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- Alternatively use the Scheduler method:</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> Sensor_data&
- V1 = *((sensor_data*) ts.currentLts());</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; text-indent: 0.49in; margin-bottom: 0in">
- …</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt"> V.distance
- = <calculate your values here>;</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <ol start="3">
- <li><p class="western" style="margin-bottom: 0in; page-break-after: avoid">
- <font face="Times New Roman, serif"><b>ENABLING TASK PRIORITIZATION
- </b></font>
- </p>
- </ol>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in; page-break-after: avoid">
- <br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">In
- certain cases you want a task to be invoked before others in case of
- scheduling collision (tasks ready to be invoked at the same time). In
- a flat execution chain scenario tasks are evaluated for execution in
- the order they were added to the chain. Therefore a single task has
- to wait for the rest of the chain to be evaluated to get a chance
- again. </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">Consider
- a scenario where a task taking gyroscope measurements has to be
- invoked as close to the actual scheduling time as possible. That is
- when task prioritization comes to help. </font></font>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">Let’s
- say tasks t4 and t5 are taking measurements from gyroscope and
- accelerometer, and tasks t1, t2 and t3 are doing something less
- important.</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Times New Roman, serif"><font size="2" style="font-size: 11pt">This
- is how such setup is coded:</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">#define
- _TASK_PRIORITY</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">#include
- <TaskScheduler.h></font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Scheduler
- r, hpr;</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">//
- Tasks</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- t1(1000, TASK_FOREVER, &tCallback, &r); //base priority</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- t2(2000, TASK_FOREVER, &tCallback, &r);</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- t3(3000, TASK_FOREVER, &tCallback, &r);</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- t4(10, TASK_FOREVER, &tCallback, &hpr); // higher priority</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Task
- t5(100, TASK_FOREVER, &tCallback, &hpr); //higher priority</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">…</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">void
- setup () {</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">
- …</p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">r.setHighPriorityScheduler(&hpr);
- </font></font>
- </p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in">
- <font face="Courier New, monospace"><font size="2" style="font-size: 10pt">r.enableAll(true);
- // this will recursively enable the higher priority tasks as well</font></font></p>
- <p class="western" style="margin-left: 0.98in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">}</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <ol start="3">
- <li><p class="western" style="margin-bottom: 0in"><font face="Times New Roman, serif"><b>FUTHER
- INFROMATION</b></font></p>
- </ol>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Please
- refer to examples, provided with TaskScheduler package for further
- information and implementation options.</font></font></p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><br>
- </p>
- <p class="western" style="margin-left: 0.49in; margin-bottom: 0in"><font face="Courier New, monospace"><font size="2" style="font-size: 10pt">Real
- time examples of TaskScheduler are available here:</font></font></p>
- <ol>
- <li><p class="western" style="margin-bottom: 0in"><font color="#0000ff"><u><a href="http://www.instructables.com/id/APIS-Automated-Plant-Irrigation-System/"><font face="Courier New, monospace">http://www.instructables.com/id/APIS-Automated-Plant-Irrigation-System/</font></a></u></font></p>
- <li><p class="western" style="margin-bottom: 0in"><font color="#0000ff"><u><a href="http://www.instructables.com/id/Wave-your-hand-to-control-OWI-Robotic-Arm-no-strin/"><font face="Courier New, monospace">http://www.instructables.com/id/Wave-your-hand-to-control-OWI-Robotic-Arm-no-strin/</font></a></u></font></p>
- <li><p class="western" style="margin-bottom: 0in"><font color="#0000ff"><u><a href="http://www.instructables.com/id/Arduino-Nano-based-Hexbug-Scarab-Robotic-Spider"><font face="Courier New, monospace">http://www.instructables.com/id/Arduino-Nano-based-Hexbug-Scarab-Robotic-Spider</font></a></u></font></p>
- </ol>
- <p class="western" style="margin-bottom: 0in"><br>
- </p>
- <div title="footer">
- <p style="margin-top: 0.35in; margin-bottom: 0in"> <sdfield type=PAGE subtype=RANDOM format=ARABIC>34</sdfield></p>
- </div>
- </body>
- </html>
|