diff options
author | Florin Coras <fcoras@cisco.com> | 2018-07-04 04:17:41 -0700 |
---|---|---|
committer | Marco Varlese <marco.varlese@suse.de> | 2018-07-06 08:40:18 +0000 |
commit | 65784c1602c7c8171effd00384f65f546d93a13b (patch) | |
tree | e649a73d3420d1df96e40d0087be6e5d1fbf1f57 /src/svm/test_svm_message_queue.c | |
parent | 01a771b12198b3f07d67a4bc73650c1f65257708 (diff) |
svm: add unidirectional message queue
Meant for single reader/writer message exchanges. Supports multiple
message rings.
Change-Id: I925de9a6ae19226c5c39a63caff76424ed123a13
Signed-off-by: Florin Coras <fcoras@cisco.com>
Diffstat (limited to 'src/svm/test_svm_message_queue.c')
-rw-r--r-- | src/svm/test_svm_message_queue.c | 176 |
1 files changed, 176 insertions, 0 deletions
diff --git a/src/svm/test_svm_message_queue.c b/src/svm/test_svm_message_queue.c new file mode 100644 index 00000000000..69ffd131ac2 --- /dev/null +++ b/src/svm/test_svm_message_queue.c @@ -0,0 +1,176 @@ +/* + * Copyright (c) 2018 Cisco and/or its affiliates. + * 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 <svm/ssvm.h> +#include <svm/message_queue.h> + +#define test1_error(_fmt, _args...) \ +{ \ + ssvm_pop_heap (oldheap); \ + error = clib_error_return (0, _fmt, ##_args); \ + goto done; \ +} + +clib_error_t * +test1 (int verbose) +{ + ssvm_private_t _ssvm, *ssvm = &_ssvm; + svm_msg_q_cfg_t _cfg, *cfg = &_cfg; + svm_msg_q_msg_t msg1, msg2, msg[12]; + ssvm_shared_header_t *sh; + clib_error_t *error = 0; + svm_msg_q_t *mq; + void *oldheap; + int i; + + memset (ssvm, 0, sizeof (*ssvm)); + + ssvm->ssvm_size = 1 << 20; + ssvm->i_am_master = 1; + ssvm->my_pid = getpid (); + ssvm->name = format (0, "%s%c", "test", 0); + ssvm->requested_va = 0; + + if (ssvm_master_init (ssvm, SSVM_SEGMENT_SHM)) + return clib_error_return (0, "failed: segment allocation"); + sh = ssvm->sh; + + svm_msg_q_ring_cfg_t rc[2]= {{8, 8, 0}, {8, 16, 0}}; + cfg->consumer_pid = ~0; + cfg->n_rings = 2; + cfg->q_nitems = 16; + cfg->ring_cfgs = rc; + + oldheap = ssvm_push_heap (sh); + mq = svm_msg_q_alloc (cfg); + if (!mq) + test1_error ("failed: alloc"); + + if (vec_len (mq->rings) != 2) + test1_error ("failed: ring allocation"); + + msg1 = svm_msg_q_alloc_msg (mq, 8); + if (mq->rings[0].cursize != 1 + || msg1.ring_index != 0 + || msg1.elt_index != 0) + test1_error ("failed: msg alloc1"); + + msg2 = svm_msg_q_alloc_msg (mq, 15); + if (mq->rings[1].cursize != 1 + || msg2.ring_index != 1 + || msg2.elt_index != 0) + test1_error ("failed: msg alloc2"); + + svm_msg_q_free_msg (mq, &msg1); + if (mq->rings[0].cursize != 0) + test1_error("failed: free msg"); + + for (i = 0; i < 12; i++) + { + msg[i] = svm_msg_q_alloc_msg (mq, 7); + *(u32 *)svm_msg_q_msg_data (mq, &msg[i]) = i; + } + + if (mq->rings[0].cursize != 8 + || mq->rings[1].cursize != 5) + test1_error ("failed: msg alloc3"); + + *(u32 *)svm_msg_q_msg_data (mq, &msg2) = 123; + svm_msg_q_add (mq, msg2, SVM_Q_NOWAIT); + for (i = 0; i < 12; i++) + svm_msg_q_add (mq, msg[i], SVM_Q_NOWAIT); + + if (svm_msg_q_sub (mq, &msg2, SVM_Q_NOWAIT, 0)) + test1_error ("failed: dequeue1"); + + if (msg2.ring_index != 1 || msg2.elt_index != 0) + test1_error ("failed: dequeue1 result"); + if (*(u32 *)svm_msg_q_msg_data (mq, &msg2) != 123) + test1_error ("failed: dequeue1 wrong data"); + + svm_msg_q_free_msg (mq, &msg2); + + for (i = 0; i < 12; i++) + { + if (svm_msg_q_sub (mq, &msg[i], SVM_Q_NOWAIT, 0)) + test1_error ("failed: dequeue2"); + if (i < 8) + { + if (msg[i].ring_index != 0 || msg[i].elt_index != (i + 1) % 8) + test1_error ("failed: dequeue2 result2"); + } + else + { + if (msg[i].ring_index != 1 || msg[i].elt_index != (i - 8) + 1) + test1_error ("failed: dequeue2 result3"); + } + if (*(u32 *)svm_msg_q_msg_data (mq, &msg[i]) != i) + test1_error ("failed: dequeue2 wrong data"); + svm_msg_q_free_msg (mq, &msg[i]); + } + if (mq->rings[0].cursize != 0 || mq->rings[1].cursize != 0) + test1_error ("failed: post dequeue"); + + ssvm_pop_heap (oldheap); + +done: + ssvm_delete (ssvm); + return error; +} + +int +test_svm_message_queue (unformat_input_t * input) +{ + clib_error_t *error = 0; + int verbose = 0; + int test_id = 0; + + while (unformat_check_input (input) != UNFORMAT_END_OF_INPUT) + { + if (unformat (input, "test1")) + test_id = 1; + else + { + error = clib_error_create ("unknown input `%U'\n", + format_unformat_error, input); + goto out; + } + } + + switch (test_id) + { + case 1: + error = test1 (verbose); + } +out: + if (error) + clib_error_report (error); + else + clib_warning ("success"); + + return 0; +} + +int +main (int argc, char *argv[]) +{ + unformat_input_t i; + int r; + + unformat_init_command_line (&i, argv); + r = test_svm_message_queue (&i); + unformat_free (&i); + return r; +} |