forked from arkhipenko/TaskScheduler
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathTaskScheduler.html
3204 lines (3197 loc) · 250 KB
/
TaskScheduler.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
<!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="20160202;231400000000000">
<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.1.0: 2016-02-02</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 a
container concept that links together:</p>
<ol>
<li><p class="western" style="margin-bottom: 0in">Program code
performing specific task 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 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"><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>