Как задача может ждать в нескольких очередях vxworks?

У нас есть дизайн vxWorks, который требует одной задачи для обработки сообщений с высоким и низким приоритетом, отправленных через две очереди сообщений.
Сообщения с заданным приоритетом должны обрабатываться в порядке FIFO.

Например, обработайте все сообщения с высоким приоритетом в порядке их получения, затем обработайте сообщения с низким приоритетом. Если сообщения с высоким приоритетом нет, немедленно обработайте сообщение с низким приоритетом.

Есть ли способ сделать это?

2 ответа

Решение

Если вы используете именованные каналы (pipeDevCreate(), write(), read()) вместо очередей сообщений, вы можете использовать select () для блокировки до тех пор, пока в каком-либо канале не появятся сообщения.

Всякий раз, когда срабатывает select (), вы обрабатываете все сообщения в канале с высоким приоритетом. Затем вы обрабатываете одно сообщение из канала с низким приоритетом. Затем снова вызовите select (loop).

Примеры фрагментов кода:

 // Initialization: Create high and low priority named pipes
 pipeDrv(); //initialize pipe driver
 int fdHi = pipeDevCreate("/pipe/high",numMsgs,msgSize);
 int fdLo = pipeDevCreate("/pipe/low",numMsgs,msgSize);

 ...

 // Message sending thread: Add messages to pipe
 write(fdHi, buf, sizeof(buf));

 ...

 // Message processing Thread: select loop
 fd_set rdFdSet;

 while(1)
 {
     FD_ZERO(&rdFdSet);
     FD_SET(fdHi, &rdFdSet);
     FD_SET(fdLo, &rdFdSet;

     if (select(FD_SETSIZE, &rdFdSet, NULL, NULL, NULL) != ERROR)
     {
         if (FD_ISSET(fdHi, &rdFdSet))
         {
             // process all high-priority messages
             while(read(fdHi,buf,size) > 0)
             {
                 //process high-priority
             }
         }

         if (FD_ISSET(fdLo, &rdFdSet))
         {
             // process a single low priority message
             if (read(fdLo,buf,size) > 0)
             {
                 // process low priority
             }
         }
     }
 }

В vxWorks вы не можете ожидать сразу нескольких очередей. Однако вы можете использовать события ОС (из eventLib) для достижения этого результата. Вот простой фрагмент кода:


MSG_Q_ID lowQ, hiQ;

void Init() {
// Task Initialization Code.  This should be called from the task that will
// be receiving the messages
...
hiQ = msgQCreate(...);
lowQ = msgQCreate(...);
msgQEvStart(hiQ, VX_EV01);  // Event 1 sent when hiQ receives message
msgQEvStart(loQ, VX_EV02);  // Event 2 sent when loQ receives message
...
}
void RxMessages() {
...
  UINT32 ev;   // Event received

   // Blocks until we receive Event 1 or 2
   eventReceive(VX_EV01 | VX_EV02, EVENT_WAIT_ANY, WAIT_FOREVER, &ev);
   if(ev & VX_EV01) {
      msgQReceive(hiQ, ...);
   }
   if(ev & VX_EV02) {
      msgQReceive(loQ, ...);
   }
}

Обратите внимание, что вам нужно изменить этот код, чтобы убедиться, что вы исчерпали все свои очереди в случае, если было получено более одного сообщения.

Тот же механизм также может быть применен к двоичным семафорам с помощью функции semEvStart().

Другие вопросы по тегам