123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245 |
- /*
- ChibiOS - Copyright (C) 2006..2018 Giovanni Di Sirio
- Licensed under the Apache License, Version 2.0 (the "License");
- you may not use this file except in compliance with the License.
- You may obtain a copy of the License at
- http://www.apache.org/licenses/LICENSE-2.0
- Unless required by applicable law or agreed to in writing, software
- distributed under the License is distributed on an "AS IS" BASIS,
- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- See the License for the specific language governing permissions and
- limitations under the License.
- */
- #include "ch.hpp"
- #include "hal.h"
- #include "rt_test_root.h"
- #include "oslib_test_root.h"
- using namespace chibios_rt;
- /*
- * Message server thread class. It receives messages and does nothing except
- * reply after the specified time.
- */
- class MessageServerThread : public BaseStaticThread<256> {
- protected:
- void main(void) override {
- setName("server");
- while (true) {
- ThreadReference sender = waitMessage();
- time_msecs_t msecs = (time_msecs_t)sender.getMessage();
- sleep(TIME_MS2I(msecs));
- sender.releaseMessage(0);
- }
- }
- public:
- MessageServerThread(void) : BaseStaticThread<256>() {
- }
- };
- /* Reference to the server thread.*/
- static ThreadReference sref;
- /*
- * LED blink sequences.
- * NOTE: Sequences must always be terminated by a GOTO instruction.
- * NOTE: The sequencer language could be easily improved but this is outside
- * the scope of this demo.
- */
- #define SLEEP 0
- #define GOTO 1
- #define STOP 2
- #define BITCLEAR 3
- #define BITSET 4
- #define MESSAGE 5
- typedef struct {
- uint8_t action;
- union {
- msg_t msg;
- uint32_t value;
- ioline_t line;
- };
- } seqop_t;
- // Flashing sequence for LED3.
- static const seqop_t LED3_sequence[] =
- {
- {BITSET, LINE_LED3},
- {SLEEP, 800},
- {BITCLEAR, LINE_LED3},
- {SLEEP, 200},
- {GOTO, 0}
- };
- // Flashing sequence for LED4.
- static const seqop_t LED4_sequence[] =
- {
- {BITSET, LINE_LED4},
- {SLEEP, 600},
- {BITCLEAR, LINE_LED4},
- {SLEEP, 400},
- {GOTO, 0}
- };
- // Flashing sequence for LED5.
- static const seqop_t LED5_sequence[] =
- {
- {BITSET, LINE_LED5},
- {SLEEP, 400},
- {BITCLEAR, LINE_LED5},
- {SLEEP, 600},
- {GOTO, 0}
- };
- // Flashing sequence for LED6.
- static const seqop_t LED6_sequence[] =
- {
- {BITSET, LINE_LED6},
- {SLEEP, 200},
- {BITCLEAR, LINE_LED6},
- {SLEEP, 800},
- {GOTO, 0}
- };
- // Message sequence.
- static const seqop_t msg_sequence[] =
- {
- {MESSAGE, 50},
- {SLEEP, 1000},
- {GOTO, 0}
- };
- /*
- * Sequencer thread class. It can drive LEDs or other output pins.
- * Any sequencer is just an instance of this class, all the details are
- * totally encapsulated and hidden to the application level.
- */
- class SequencerThread : public BaseStaticThread<128> {
- private:
- const seqop_t *base, *curr; // Thread local variables.
- protected:
- void main(void) override {
- setName("sequencer");
- while (true) {
- switch(curr->action) {
- case SLEEP:
- sleep(TIME_MS2I(curr->value));
- break;
- case GOTO:
- curr = &base[curr->value];
- continue;
- case STOP:
- return;
- case BITCLEAR:
- palClearLine(curr->line);
- break;
- case BITSET:
- palSetLine(curr->line);
- break;
- case MESSAGE:
- sref.sendMessage(curr->msg);
- break;
- }
- curr++;
- }
- }
- public:
- SequencerThread(const seqop_t *sequence) : BaseStaticThread<128>() {
- base = curr = sequence;
- }
- };
- /*
- * Tester thread class. This thread executes the test suite.
- */
- class TesterThread : public BaseStaticThread<256> {
- protected:
- void main(void) override {
- setName("tester");
- test_execute((BaseSequentialStream *)&SD2, &rt_test_suite);
- test_execute((BaseSequentialStream *)&SD2, &oslib_test_suite);
- exit(test_global_fail);
- }
- public:
- TesterThread(void) : BaseStaticThread<256>() {
- }
- };
- /* Static threads instances.*/
- static TesterThread tester;
- static MessageServerThread server_thread;
- static SequencerThread blinker1(LED3_sequence);
- static SequencerThread blinker2(LED4_sequence);
- static SequencerThread blinker3(LED5_sequence);
- static SequencerThread blinker4(LED6_sequence);
- static SequencerThread sender1(msg_sequence);
- /*
- * Application entry point.
- */
- int main(void) {
- /*
- * System initializations.
- * - HAL initialization, this also initializes the configured device drivers
- * and performs the board-specific initializations.
- * - Kernel initialization, the main() function becomes a thread and the
- * RTOS is active.
- */
- halInit();
- System::init();
- /*
- * Activates the serial driver 2 using the driver default configuration.
- * PA2(TX) and PA3(RX) are routed to USART2.
- */
- sdStart(&SD2, NULL);
- palSetPadMode(GPIOA, 2, PAL_MODE_ALTERNATE(7));
- palSetPadMode(GPIOA, 3, PAL_MODE_ALTERNATE(7));
- /*
- * Starting the message server thread, storing the returned reference.
- */
- sref = server_thread.start(NORMALPRIO + 20);
- /*
- * Starts several instances of the SequencerThread class, each one operating
- * on a different sequence.
- */
- blinker1.start(NORMALPRIO + 10);
- blinker2.start(NORMALPRIO + 10);
- blinker3.start(NORMALPRIO + 10);
- blinker4.start(NORMALPRIO + 10);
- sender1.start(NORMALPRIO + 10);
- /*
- * Serves timer events.
- */
- while (true) {
- if (palReadPad(GPIOA, GPIOA_BUTTON)) {
- ThreadReference tref = tester.start(NORMALPRIO);
- tref.wait();
- };
- BaseThread::sleep(TIME_MS2I(500));
- }
- return 0;
- }
|