/* Copyright (c) 2013 The Chromium OS Authors. All rights reserved. * Use of this source code is governed by a BSD-style license that can be * found in the LICENSE file. * * Test queue. */ #include "common.h" #include "console.h" #include "queue.h" #include "test_util.h" #include "timer.h" #include "util.h" static struct queue const test_queue8 = QUEUE_NULL(8, char); static struct queue const test_queue2 = QUEUE_NULL(2, int16_t); static int test_queue8_empty(void) { char dummy = 1; queue_init(&test_queue8); TEST_ASSERT(queue_is_empty(&test_queue8)); TEST_ASSERT(!queue_remove_units(&test_queue8, &dummy, 1)); TEST_ASSERT(queue_add_units(&test_queue8, &dummy, 1) == 1); TEST_ASSERT(!queue_is_empty(&test_queue8)); return EC_SUCCESS; } static int test_queue8_init(void) { char dummy = 1; queue_init(&test_queue8); TEST_ASSERT(queue_add_units(&test_queue8, &dummy, 1) == 1); queue_init(&test_queue8); TEST_ASSERT(queue_is_empty(&test_queue8)); return EC_SUCCESS; } static int test_queue8_fifo(void) { char buf1[3] = {1, 2, 3}; char buf2[3]; queue_init(&test_queue8); TEST_ASSERT(queue_add_units(&test_queue8, buf1 + 0, 1) == 1); TEST_ASSERT(queue_add_units(&test_queue8, buf1 + 1, 1) == 1); TEST_ASSERT(queue_add_units(&test_queue8, buf1 + 2, 1) == 1); TEST_ASSERT(queue_remove_units(&test_queue8, buf2, 3) == 3); TEST_ASSERT_ARRAY_EQ(buf1, buf2, 3); return EC_SUCCESS; } static int test_queue8_multiple_units_add(void) { char buf1[5] = {1, 2, 3, 4, 5}; char buf2[5]; queue_init(&test_queue8); TEST_ASSERT(queue_space(&test_queue8) >= 5); TEST_ASSERT(queue_add_units(&test_queue8, buf1, 5) == 5); TEST_ASSERT(queue_remove_units(&test_queue8, buf2, 5) == 5); TEST_ASSERT_ARRAY_EQ(buf1, buf2, 5); return EC_SUCCESS; } static int test_queue8_removal(void) { char buf1[5] = {1, 2, 3, 4, 5}; char buf2[5]; queue_init(&test_queue8); TEST_ASSERT(queue_add_units(&test_queue8, buf1, 5) == 5); /* 1, 2, 3, 4, 5 */ TEST_ASSERT(queue_remove_units(&test_queue8, buf2, 3) == 3); TEST_ASSERT_ARRAY_EQ(buf1, buf2, 3); /* 4, 5 */ TEST_ASSERT(queue_add_units(&test_queue8, buf1, 2) == 2); /* 4, 5, 1, 2 */ TEST_ASSERT(queue_space(&test_queue8) == 4); TEST_ASSERT(queue_remove_units(&test_queue8, buf2, 1) == 1); TEST_ASSERT(buf2[0] == 4); /* 5, 1, 2 */ TEST_ASSERT(queue_add_units(&test_queue8, buf1 + 2, 2) == 2); /* 5, 1, 2, 3, 4 */ TEST_ASSERT(queue_space(&test_queue8) == 3); TEST_ASSERT(queue_add_units(&test_queue8, buf1 + 2, 3) == 3); /* 5, 1, 2, 3, 4, 3, 4, 5 */ TEST_ASSERT(queue_space(&test_queue8) == 0); TEST_ASSERT(queue_remove_units(&test_queue8, buf2, 1) == 1); TEST_ASSERT(buf2[0] == 5); TEST_ASSERT(queue_remove_units(&test_queue8, buf2, 4) == 4); TEST_ASSERT_ARRAY_EQ(buf1, buf2, 4); TEST_ASSERT(queue_remove_units(&test_queue8, buf2, 3) == 3); TEST_ASSERT_ARRAY_EQ(buf1 + 2, buf2, 3); TEST_ASSERT(queue_is_empty(&test_queue8)); /* Empty */ TEST_ASSERT(queue_add_units(&test_queue8, buf1, 5) == 5); TEST_ASSERT(queue_remove_units(&test_queue8, buf2, 5) == 5); TEST_ASSERT_ARRAY_EQ(buf1, buf2, 5); return EC_SUCCESS; } static int test_queue8_peek(void) { char buf1[5] = {1, 2, 3, 4, 5}; char buf2[5]; queue_init(&test_queue8); TEST_ASSERT(queue_add_units(&test_queue8, buf1, 5) == 5); /* 1, 2, 3, 4, 5 */ TEST_ASSERT(queue_count(&test_queue8) == 5); TEST_ASSERT(queue_space(&test_queue8) == 3); TEST_ASSERT(queue_peek_units(&test_queue8, buf2, 2, 3) == 3); TEST_ASSERT_ARRAY_EQ(buf1 + 2, buf2, 3); TEST_ASSERT(queue_count(&test_queue8) == 5); TEST_ASSERT(queue_space(&test_queue8) == 3); return EC_SUCCESS; } static int test_queue2_odd_even(void) { uint16_t buf1[3] = {1, 2, 3}; uint16_t buf2[3]; queue_init(&test_queue2); TEST_ASSERT(queue_add_units(&test_queue2, buf1, 1) == 1); /* 1 */ TEST_ASSERT(queue_space(&test_queue2) == 1); TEST_ASSERT(queue_add_units(&test_queue2, buf1 + 1, 1) == 1); /* 1, 2 */ TEST_ASSERT(queue_space(&test_queue2) == 0); TEST_ASSERT(queue_remove_units(&test_queue2, buf2, 2) == 2); TEST_ASSERT_ARRAY_EQ(buf1, buf2, 2); TEST_ASSERT(queue_is_empty(&test_queue2)); /* Empty */ TEST_ASSERT(queue_space(&test_queue2) == 2); TEST_ASSERT(queue_add_units(&test_queue2, buf1 + 2, 1) == 1); /* 3 */ TEST_ASSERT(queue_remove_units(&test_queue2, buf2, 1) == 1); TEST_ASSERT(buf2[0] == 3); TEST_ASSERT(queue_is_empty(&test_queue2)); return EC_SUCCESS; } static int test_queue8_chunks(void) { static uint8_t const data[3] = {1, 2, 3}; struct queue_chunk chunk; queue_init(&test_queue8); chunk = queue_get_write_chunk(&test_queue8); TEST_ASSERT(chunk.length == 8); memcpy(chunk.buffer, data, 3); TEST_ASSERT(queue_advance_tail(&test_queue8, 3) == 3); chunk = queue_get_read_chunk(&test_queue8); TEST_ASSERT(chunk.length == 3); TEST_ASSERT_ARRAY_EQ(chunk.buffer, data, 3); TEST_ASSERT(queue_advance_head(&test_queue8, 3) == 3); TEST_ASSERT(queue_is_empty(&test_queue8)); return EC_SUCCESS; } static int test_queue8_chunks_wrapped(void) { static uint8_t const data[3] = {1, 2, 3}; queue_init(&test_queue8); /* Move near the end of the queue */ TEST_ASSERT(queue_advance_tail(&test_queue8, 6) == 6); TEST_ASSERT(queue_advance_head(&test_queue8, 6) == 6); /* Add three units, causing the tail to wrap */ TEST_ASSERT(queue_add_units(&test_queue8, data, 3) == 3); /* * With a wrapped tail we should only be able to access the first two * elements for reading, but all five free elements for writing. */ TEST_ASSERT(queue_get_read_chunk(&test_queue8).length == 2); TEST_ASSERT(queue_get_write_chunk(&test_queue8).length == 5); /* Signal that we have read an element */ TEST_ASSERT(queue_advance_head(&test_queue8, 1) == 1); /* * Now we should only be able to see a single element for reading, but * all six free element. */ TEST_ASSERT(queue_get_read_chunk(&test_queue8).length == 1); TEST_ASSERT(queue_get_write_chunk(&test_queue8).length == 6); /* Signal that we have read the last two elements */ TEST_ASSERT(queue_advance_head(&test_queue8, 2) == 2); /* * Now there should be no elements available for reading, and only * seven, not eight elements available for writing. This is because * the head/tail pointers now point to the second unit in the array. */ TEST_ASSERT(queue_get_read_chunk(&test_queue8).length == 0); TEST_ASSERT(queue_get_write_chunk(&test_queue8).length == 7); return EC_SUCCESS; } static int test_queue8_chunks_full(void) { static uint8_t const data[8] = {1, 2, 3, 4, 5, 6, 7, 8}; struct queue_chunk chunk; queue_init(&test_queue8); /* Move near the end of the queue */ TEST_ASSERT(queue_advance_tail(&test_queue8, 6) == 6); TEST_ASSERT(queue_advance_head(&test_queue8, 6) == 6); /* Fill the queue */ TEST_ASSERT(queue_add_units(&test_queue8, data, 8) == 8); /* With a full queue we shouldn't be able to write */ TEST_ASSERT(queue_get_write_chunk(&test_queue8).length == 0); /* But we should be able to read, though only two entries at first */ chunk = queue_get_read_chunk(&test_queue8); TEST_ASSERT(chunk.length == 2); TEST_ASSERT_ARRAY_EQ(chunk.buffer, data, 2); /* Signal that we have read both units */ TEST_ASSERT(queue_advance_head(&test_queue8, 2) == 2); /* Now we should only be able to see the rest */ chunk = queue_get_read_chunk(&test_queue8); TEST_ASSERT(chunk.length == 6); TEST_ASSERT_ARRAY_EQ(chunk.buffer, data + 2, 6); return EC_SUCCESS; } static int test_queue8_chunks_empty(void) { queue_init(&test_queue8); /* With an empty queue we shouldn't be able to read */ TEST_ASSERT(queue_get_read_chunk(&test_queue8).length == 0); /* But we should be able to write, everything */ TEST_ASSERT(queue_get_write_chunk(&test_queue8).length == 8); return EC_SUCCESS; } static int test_queue8_chunks_advance(void) { queue_init(&test_queue8); /* * We should only be able to advance the tail (add units) as many * units as there are in an empty queue. */ TEST_ASSERT(queue_advance_tail(&test_queue8, 10) == 8); /* * Similarly, we should only be able to advance the head (remove * units) as many units as there are in the now full queue. */ TEST_ASSERT(queue_advance_head(&test_queue8, 10) == 8); /* * And it shouldn't matter if we start in the middle of the queue. */ TEST_ASSERT(queue_advance_tail(&test_queue8, 3) == 3); TEST_ASSERT(queue_advance_head(&test_queue8, 3) == 3); TEST_ASSERT(queue_advance_tail(&test_queue8, 10) == 8); TEST_ASSERT(queue_advance_head(&test_queue8, 10) == 8); return EC_SUCCESS; } void run_test(void) { test_reset(); RUN_TEST(test_queue8_empty); RUN_TEST(test_queue8_init); RUN_TEST(test_queue8_fifo); RUN_TEST(test_queue8_multiple_units_add); RUN_TEST(test_queue8_removal); RUN_TEST(test_queue8_peek); RUN_TEST(test_queue2_odd_even); RUN_TEST(test_queue8_chunks); RUN_TEST(test_queue8_chunks_wrapped); RUN_TEST(test_queue8_chunks_full); RUN_TEST(test_queue8_chunks_empty); RUN_TEST(test_queue8_chunks_advance); test_print_result(); }