Repository: incubator-mynewt-core
Updated Branches:
  refs/heads/develop 29b69ddf2 -> c483e8222


eventq - os_eventq_poll() - Allow a timeout of 0.

If the timeout is 0, don't involve the scheduler at all.  Grab an event
if one is available, else return immediately.


Project: http://git-wip-us.apache.org/repos/asf/incubator-mynewt-core/repo
Commit: 
http://git-wip-us.apache.org/repos/asf/incubator-mynewt-core/commit/24b90c33
Tree: http://git-wip-us.apache.org/repos/asf/incubator-mynewt-core/tree/24b90c33
Diff: http://git-wip-us.apache.org/repos/asf/incubator-mynewt-core/diff/24b90c33

Branch: refs/heads/develop
Commit: 24b90c33a4bc6fb10110d3218e372d24dcff2ed9
Parents: 29b69dd
Author: Christopher Collins <[email protected]>
Authored: Thu Aug 4 12:41:46 2016 -0700
Committer: Christopher Collins <[email protected]>
Committed: Thu Aug 4 12:41:46 2016 -0700

----------------------------------------------------------------------
 libs/os/src/os_eventq.c        |  33 +++++++-
 libs/os/src/test/eventq_test.c | 149 +++++++++++++++++++++++-------------
 2 files changed, 125 insertions(+), 57 deletions(-)
----------------------------------------------------------------------


http://git-wip-us.apache.org/repos/asf/incubator-mynewt-core/blob/24b90c33/libs/os/src/os_eventq.c
----------------------------------------------------------------------
diff --git a/libs/os/src/os_eventq.c b/libs/os/src/os_eventq.c
index b3ff26f..f9cc283 100644
--- a/libs/os/src/os_eventq.c
+++ b/libs/os/src/os_eventq.c
@@ -117,6 +117,29 @@ pull_one:
     return (ev);
 }
 
+static struct os_event *
+os_eventq_poll_0timo(struct os_eventq **evq, int nevqs)
+{
+    struct os_event *ev;
+    os_sr_t sr;
+    int i;
+
+    ev = NULL;
+
+    OS_ENTER_CRITICAL(sr);
+    for (i = 0; i < nevqs; i++) {
+        ev = STAILQ_FIRST(&evq[i]->evq_list);
+        if (ev) {
+            STAILQ_REMOVE(&evq[i]->evq_list, ev, os_event, ev_next);
+            ev->ev_queued = 0;
+            break;
+        }
+    }
+    OS_EXIT_CRITICAL(sr);
+
+    return ev;
+}
+
 /**
  * Poll the list of event queues specified by the evq parameter 
  * (size nevqs), and return the "first" event available on any of 
@@ -137,6 +160,13 @@ os_eventq_poll(struct os_eventq **evq, int nevqs, 
os_time_t timo)
     int i, j;
     os_sr_t sr;
 
+    /* If the timeout is 0, don't involve the scheduler at all.  Grab an event
+     * if one is available, else return immediately.
+     */
+    if (timo == 0) {
+        return os_eventq_poll_0timo(evq, nevqs);
+    }
+
     ev = NULL;
 
     OS_ENTER_CRITICAL(sr);
@@ -147,8 +177,7 @@ os_eventq_poll(struct os_eventq **evq, int nevqs, os_time_t 
timo)
         if (ev) {
             STAILQ_REMOVE(&evq[i]->evq_list, ev, os_event, ev_next);
             ev->ev_queued = 0;
-            /* Reset the items that already have an evq task set
-             */
+            /* Reset the items that already have an evq task set. */
             for (j = 0; j < i; j++) {
                 evq[j]->evq_task = NULL;
             }

http://git-wip-us.apache.org/repos/asf/incubator-mynewt-core/blob/24b90c33/libs/os/src/test/eventq_test.c
----------------------------------------------------------------------
diff --git a/libs/os/src/test/eventq_test.c b/libs/os/src/test/eventq_test.c
index f17e5e4..cb1ed94 100644
--- a/libs/os/src/test/eventq_test.c
+++ b/libs/os/src/test/eventq_test.c
@@ -17,7 +17,7 @@
  * under the License.
  */
  
-
+#include <string.h>
 #include "testutil/testutil.h"
 #include "os/os.h"
 #include "os_test_priv.h"
@@ -59,10 +59,6 @@ os_stack_t eventq_task_stack_poll_s[POLL_STACK_SIZE];
 struct os_task eventq_task_poll_r;
 os_stack_t eventq_task_stack_poll_r[POLL_STACK_SIZE ];
 
-struct os_eventq *poll_eventq[POLL_STACK_SIZE];
-struct os_eventq p_event[POLL_STACK_SIZE];
-struct os_event poll_event[POLL_STACK_SIZE];
-
 /* Setting the data for the poll timeout */
 /* Define the task stack for the eventq_task_poll_timeout_send */
 #define SEND_TASK_POLL_TIMEOUT_PRIO        (5)
@@ -74,10 +70,6 @@ os_stack_t eventq_task_stack_poll_timeout_s[POLL_STACK_SIZE];
 struct os_task eventq_task_poll_timeout_r;
 os_stack_t eventq_task_stack_poll_timeout_r[POLL_STACK_SIZE];
 
-struct os_eventq *poll_eventq_timeout[POLL_STACK_SIZE];
-struct os_eventq p_event_timeout[POLL_STACK_SIZE];
-struct os_event poll_event_timeout[POLL_STACK_SIZE];
-
 /* Setting the data for the poll single */
 /* Define the task stack for the eventq_task_poll_single_send */
 #define SEND_TASK_POLL_SINGLE_PRIO        (7)
@@ -89,10 +81,6 @@ os_stack_t eventq_task_stack_poll_single_s[POLL_STACK_SIZE];
 struct os_task eventq_task_poll_single_r;
 os_stack_t eventq_task_stack_poll_single_r[POLL_STACK_SIZE];
 
-struct os_eventq *poll_eventq_single[POLL_STACK_SIZE];
-struct os_eventq p_event_single[POLL_STACK_SIZE];
-struct os_event poll_event_single[POLL_STACK_SIZE];
-
 /* This is the task function  to send data */
 void
 eventq_task_send(void *arg)
@@ -143,24 +131,19 @@ eventq_task_receive(void *arg)
 void
 eventq_task_poll_send(void *arg)
 {
+    struct os_eventq *eventqs[SIZE_MULTI_EVENT];
     int i;
 
-    /* Assigning the *poll_eventq[] */
     for (i = 0; i < SIZE_MULTI_EVENT; i++){
-        poll_eventq[i] = &p_event[i];
+        eventqs[i] = &multi_eventq[i];
     }
 
-    /* Initializing the *poll_eventq */
     for (i = 0; i < SIZE_MULTI_EVENT; i++){
-        os_eventq_init(poll_eventq[i]);
-    }
-
-    for (i = 0; i < SIZE_MULTI_EVENT; i++){
-        poll_event[i].ev_type = i + 10;
-        poll_event[i].ev_arg = NULL;
+        m_event[i].ev_type = i + 10;
+        m_event[i].ev_arg = NULL;
 
         /* Put and send */
-        os_eventq_put(poll_eventq[i], &poll_event[i]);
+        os_eventq_put(eventqs[i], &m_event[i]);
         os_time_delay(OS_TICKS_PER_SEC / 2);
     }
 
@@ -171,12 +154,17 @@ eventq_task_poll_send(void *arg)
 void
 eventq_task_poll_receive(void *arg)
 {
+    struct os_eventq *eventqs[SIZE_MULTI_EVENT];
     struct os_event *event;
     int i;
 
+    for (i = 0; i < SIZE_MULTI_EVENT; i++){
+        eventqs[i] = &multi_eventq[i];
+    }
+
     /* Recieving using the os_eventq_poll*/
     for (i = 0; i < SIZE_MULTI_EVENT; i++) {
-        event = os_eventq_poll(poll_eventq, SIZE_MULTI_EVENT, OS_WAIT_FOREVER);
+        event = os_eventq_poll(eventqs, SIZE_MULTI_EVENT, OS_WAIT_FOREVER);
         TEST_ASSERT(event->ev_type == i +10);
     }
 
@@ -189,26 +177,18 @@ eventq_task_poll_receive(void *arg)
 void
 eventq_task_poll_timeout_send(void *arg)
 {
+    struct os_eventq *eventqs[SIZE_MULTI_EVENT];
     int i;
 
-    /* Assigning the *poll_eventq_timeout[] */
-    for (i = 0; i < SIZE_MULTI_EVENT; i++){
-        poll_eventq_timeout[i] = &p_event_timeout[i];
-    }
-
-    /* Initializing the *poll_eventq_timeout */
     for (i = 0; i < SIZE_MULTI_EVENT; i++){
-        os_eventq_init(poll_eventq_timeout[i]);
+        eventqs[i] = &multi_eventq[i];
     }
 
     for (i = 0; i < SIZE_MULTI_EVENT; i++){
          os_time_delay(1000);
 
-        poll_event_timeout[i].ev_type = i + 10;
-        poll_event_timeout[i].ev_arg = NULL;
-
         /* Put and send */
-        os_eventq_put(poll_eventq_timeout[i], &poll_event_timeout[i]);
+        os_eventq_put(eventqs[i], &m_event[i]);
         os_time_delay(OS_TICKS_PER_SEC / 2);
     }
 
@@ -221,12 +201,17 @@ eventq_task_poll_timeout_send(void *arg)
 void
 eventq_task_poll_timeout_receive(void *arg)
 {
-     struct os_event *event;
+    struct os_eventq *eventqs[SIZE_MULTI_EVENT];
+    struct os_event *event;
     int i;
 
+    for (i = 0; i < SIZE_MULTI_EVENT; i++){
+        eventqs[i] = &multi_eventq[i];
+    }
+
     /* Recieving using the os_eventq_poll_timeout*/
     for (i = 0; i < SIZE_MULTI_EVENT; i++) {
-        event = os_eventq_poll(poll_eventq_timeout, SIZE_MULTI_EVENT, 200);
+        event = os_eventq_poll(eventqs, SIZE_MULTI_EVENT, 200);
         TEST_ASSERT(event == NULL);
     }
 
@@ -239,45 +224,40 @@ eventq_task_poll_timeout_receive(void *arg)
 void
 eventq_task_poll_single_send(void *arg)
 {
+    struct os_eventq *eventqs[SIZE_MULTI_EVENT];
     int i;
     int position = 2;
 
-    /* Assigning the *poll_eventq_single */
     for (i = 0; i < SIZE_MULTI_EVENT; i++){
-        poll_eventq_single[i] = &p_event_single[i];
+        eventqs[i] = &multi_eventq[i];
     }
 
-    /* Initializing the *poll_eventq_single */
-    for (i = 0; i < SIZE_MULTI_EVENT; i++){
-        os_eventq_init(poll_eventq_single[i]);
-    }
-
-    poll_event_single[position].ev_type = 20;
-    poll_event_single[position].ev_arg = NULL;
-
-        /* Put and send */
-        os_eventq_put(poll_eventq_single[position],
-            &poll_event_single[position]);
-        os_time_delay(OS_TICKS_PER_SEC / 2);
+    /* Put and send */
+    os_eventq_put(eventqs[position], &m_event[position]);
+    os_time_delay(OS_TICKS_PER_SEC / 2);
 
     /* This task sleeps until the receive task completes the test. */
     os_time_delay(1000000);
-
 }
 
 /* Recieving the single event */
 void
 eventq_task_poll_single_receive(void *arg)
 {
+    struct os_eventq *eventqs[SIZE_MULTI_EVENT];
     struct os_event *event;
+    int i;
+
+    for (i = 0; i < SIZE_MULTI_EVENT; i++){
+        eventqs[i] = &multi_eventq[i];
+    }
 
     /* Recieving using the os_eventq_poll*/
-    event = os_eventq_poll(poll_eventq_single, SIZE_MULTI_EVENT,  
OS_WAIT_FOREVER);
+    event = os_eventq_poll(eventqs, SIZE_MULTI_EVENT, OS_WAIT_FOREVER);
     TEST_ASSERT(event->ev_type == 20);
 
     /* Finishes the test when OS has been started */
     os_test_restart();
-
 }
 
 TEST_CASE(event_test_sr)
@@ -309,6 +289,8 @@ TEST_CASE(event_test_sr)
 /* To test for the basic function of os_eventq_poll() */
 TEST_CASE(event_test_poll_sr)
 {
+    int i;
+
     /* Initializing the OS */
     os_init();
     /* Initialize the task */
@@ -321,6 +303,11 @@ TEST_CASE(event_test_poll_sr)
         NULL, RECEIVE_TASK_POLL_PRIO, OS_WAIT_FOREVER, 
eventq_task_stack_poll_r,
         POLL_STACK_SIZE);
 
+    /* Initializing the eventqs. */
+    for (i = 0; i < SIZE_MULTI_EVENT; i++){
+        os_eventq_init(&multi_eventq[i]);
+    }
+
     /* Does not return until OS_restart is called */
     os_start();
 
@@ -329,6 +316,8 @@ TEST_CASE(event_test_poll_sr)
 /* Test case for poll timeout */
 TEST_CASE(event_test_poll_timeout_sr)
 {
+    int i;
+
     /* Initializing the OS */
     os_init();
     /* Initialize the task */
@@ -341,6 +330,14 @@ TEST_CASE(event_test_poll_timeout_sr)
         eventq_task_poll_timeout_receive, NULL, RECEIVE_TASK_POLL_TIMEOUT_PRIO,
         OS_WAIT_FOREVER, eventq_task_stack_poll_timeout_r, POLL_STACK_SIZE);
 
+    /* Initializing the eventqs. */
+    for (i = 0; i < SIZE_MULTI_EVENT; i++){
+        os_eventq_init(&multi_eventq[i]);
+
+        m_event[i].ev_type = i + 10;
+        m_event[i].ev_arg = NULL;
+    }
+
     /* Does not return until OS_restart is called */
     os_start();
 
@@ -350,6 +347,8 @@ TEST_CASE(event_test_poll_timeout_sr)
 /* Test case for poll timeout */
 TEST_CASE(event_test_poll_single_sr)
 {
+    int i;
+
     /* Initializing the OS */
     os_init();
     /* Initialize the task */
@@ -362,16 +361,56 @@ TEST_CASE(event_test_poll_single_sr)
         eventq_task_poll_single_receive, NULL, RECEIVE_TASK_POLL_SINGLE_PRIO,
         OS_WAIT_FOREVER, eventq_task_stack_poll_single_r, POLL_STACK_SIZE);
 
+    for (i = 0; i < SIZE_MULTI_EVENT; i++){
+        os_eventq_init(&multi_eventq[i]);
+
+        m_event[i].ev_type = 10 * i;
+        m_event[i].ev_arg = NULL;
+    }
+
     /* Does not return until OS_restart is called */
     os_start();
 
 }
 
+/**
+ * Tests eventq_poll() with a timeout of 0.  This should not involve the
+ * scheduler at all, so it should work without starting the OS.
+ */
+TEST_CASE(event_test_poll_0timo)
+{
+    struct os_eventq *eventqs[SIZE_MULTI_EVENT];
+    struct os_event *evp;
+    struct os_event ev;
+    int i;
+
+    for (i = 0; i < SIZE_MULTI_EVENT; i++){
+        os_eventq_init(&multi_eventq[i]);
+        eventqs[i] = &multi_eventq[i];
+    }
+
+    evp = os_eventq_poll(eventqs, SIZE_MULTI_EVENT, 0);
+    TEST_ASSERT(evp == NULL);
+
+    /* Ensure no eventq thinks a task is waiting on it. */
+    for (i = 0; i < SIZE_MULTI_EVENT; i++) {
+        TEST_ASSERT(eventqs[i]->evq_task == NULL);
+    }
+
+    /* Put an event on one of the queues. */
+    memset(&ev, 0, sizeof ev);
+    ev.ev_type = 1;
+    os_eventq_put(eventqs[3], &ev);
+
+    evp = os_eventq_poll(eventqs, SIZE_MULTI_EVENT, 0);
+    TEST_ASSERT(evp == &ev);
+}
+
 TEST_SUITE(os_eventq_test_suite)
 {
     event_test_sr();
     event_test_poll_sr();
     event_test_poll_timeout_sr();
     event_test_poll_single_sr();
-
+    event_test_poll_0timo();
 }

Reply via email to