/* Copyright 2013 The ChromiumOS Authors * 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 tmp = 1; TEST_ASSERT(queue_is_empty(&test_queue8)); TEST_ASSERT(!queue_remove_units(&test_queue8, &tmp, 1)); TEST_ASSERT(queue_add_units(&test_queue8, &tmp, 1) == 1); TEST_ASSERT(!queue_is_empty(&test_queue8)); return EC_SUCCESS; } static int test_queue8_init(void) { char tmp = 1; TEST_ASSERT(queue_add_units(&test_queue8, &tmp, 1) == 1); queue_init(&test_queue8); TEST_ASSERT(queue_is_empty(&test_queue8)); TEST_ASSERT(queue_remove_unit(&test_queue8, &tmp) == 0); return EC_SUCCESS; } static int test_queue8_fifo(void) { char buf1[3] = { 1, 2, 3 }; char buf2[3]; 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[] = { 1, 2, 3, 4, 5, 6, 7, 8, 9 }; char buf2[5]; 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); TEST_ASSERT(queue_add_units(&test_queue8, buf1, 8) == 8); TEST_ASSERT(queue_add_unit(&test_queue8, &buf1[8]) == 0); return EC_SUCCESS; } static int test_queue8_removal(void) { char buf1[5] = { 1, 2, 3, 4, 5 }; char buf2[5]; 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]; 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]; 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; chunk = queue_get_write_chunk(&test_queue8, 0); TEST_ASSERT(chunk.count == 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.count == 3); TEST_ASSERT_ARRAY_EQ((uint8_t *)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 }; /* 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).count == 2); TEST_ASSERT(queue_get_write_chunk(&test_queue8, 0).count == 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).count == 1); TEST_ASSERT(queue_get_write_chunk(&test_queue8, 0).count == 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).count == 0); TEST_ASSERT(queue_get_write_chunk(&test_queue8, 0).count == 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; /* 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, 0).count == 0); /* But we should be able to read, though only two entries at first */ chunk = queue_get_read_chunk(&test_queue8); TEST_ASSERT(chunk.count == 2); TEST_ASSERT_ARRAY_EQ((uint8_t *)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.count == 6); TEST_ASSERT_ARRAY_EQ((uint8_t *)chunk.buffer, data + 2, 6); return EC_SUCCESS; } static int test_queue8_chunks_empty(void) { /* With an empty queue we shouldn't be able to read */ TEST_ASSERT(queue_get_read_chunk(&test_queue8).count == 0); /* But we should be able to write, everything */ TEST_ASSERT(queue_get_write_chunk(&test_queue8, 0).count == 8); return EC_SUCCESS; } static int test_queue8_chunks_advance(void) { /* * 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; } static int test_queue8_chunks_offset(void) { /* Check offsetting by 1 */ TEST_ASSERT(queue_get_write_chunk(&test_queue8, 1).count == 7); TEST_ASSERT(queue_get_write_chunk(&test_queue8, 1).buffer == test_queue8.buffer + 1); /* Check offsetting by 4 */ TEST_ASSERT(queue_get_write_chunk(&test_queue8, 4).count == 4); TEST_ASSERT(queue_get_write_chunk(&test_queue8, 4).buffer == test_queue8.buffer + 4); /* Check offset wrapping around */ TEST_ASSERT(queue_get_write_chunk(&test_queue8, 10).count == 0); TEST_ASSERT(queue_get_write_chunk(&test_queue8, 10).buffer == NULL); /* * Check offsetting when used memory is in the middle: * H T * |--xx----| */ TEST_ASSERT(queue_advance_tail(&test_queue8, 4) == 4); TEST_ASSERT(queue_advance_head(&test_queue8, 2) == 2); /* Get writable chunk to right of tail. */ TEST_ASSERT(queue_get_write_chunk(&test_queue8, 2).count == 2); TEST_ASSERT(queue_get_write_chunk(&test_queue8, 2).buffer == test_queue8.buffer + 6); /* Get writable chunk wrapped and before head. */ TEST_ASSERT(queue_get_write_chunk(&test_queue8, 4).count == 2); TEST_ASSERT(queue_get_write_chunk(&test_queue8, 4).buffer == test_queue8.buffer); /* Check offsetting into non-writable memory. */ TEST_ASSERT(queue_get_write_chunk(&test_queue8, 6).count == 0); TEST_ASSERT(queue_get_write_chunk(&test_queue8, 6).buffer == NULL); return EC_SUCCESS; } static int test_queue8_iterate_begin(void) { struct queue const *q = &test_queue8; char data[8] = { 0, 1, 2, 3, 4, 5, 6, 7 }; struct queue_iterator it; queue_begin(q, &it); TEST_EQ(it.ptr, NULL, "%p"); queue_add_units(q, data, 4); queue_begin(q, &it); TEST_EQ(*((char *)it.ptr), 0, "%d"); return EC_SUCCESS; } static int test_queue8_iterate_next(void) { struct queue const *q = &test_queue8; char data[8] = { 0, 1, 2, 3, 4, 5, 6, 7 }; struct queue_iterator it; queue_add_units(q, data, 4); queue_begin(q, &it); TEST_EQ(*((char *)it.ptr), 0, "%d"); queue_next(q, &it); TEST_NE(it.ptr, NULL, "%p"); TEST_EQ(*((char *)it.ptr), 1, "%d"); queue_next(q, &it); TEST_NE(it.ptr, NULL, "%p"); TEST_EQ(*((char *)it.ptr), 2, "%d"); queue_next(q, &it); TEST_NE(it.ptr, NULL, "%p"); TEST_EQ(*((char *)it.ptr), 3, "%d"); queue_next(q, &it); TEST_EQ(it.ptr, NULL, "%p"); return EC_SUCCESS; } static int test_queue2_iterate_next_full(void) { struct queue const *q = &test_queue2; int16_t data[2] = { 523, -788 }; struct queue_iterator it; queue_add_units(q, data, 2); queue_begin(q, &it); TEST_EQ(*((int16_t *)it.ptr), 523, "%d"); queue_next(q, &it); TEST_NE(it.ptr, NULL, "%p"); TEST_EQ(*((int16_t *)it.ptr), -788, "%d"); queue_next(q, &it); TEST_EQ(it.ptr, NULL, "%p"); queue_next(q, &it); TEST_EQ(it.ptr, NULL, "%p"); return EC_SUCCESS; } static int test_queue8_iterate_next_reset_on_change(void) { struct queue const *q = &test_queue8; char data[8] = { -88, -37, -5, -1, 3, 16, 56, 100 }; struct queue_iterator it; queue_add_units(q, data, 4); queue_begin(q, &it); TEST_NE(it.ptr, NULL, "%p"); queue_add_units(q, data + 4, 4); queue_next(q, &it); TEST_EQ(it.ptr, NULL, "%p"); queue_begin(q, &it); TEST_NE(it.ptr, NULL, "%p"); queue_advance_head(q, 3); queue_next(q, &it); TEST_EQ(it.ptr, NULL, "%p"); return EC_SUCCESS; } void before_test(void) { queue_init(&test_queue2); queue_init(&test_queue8); } void run_test(int argc, const char **argv) { 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); RUN_TEST(test_queue8_chunks_offset); RUN_TEST(test_queue8_iterate_begin); RUN_TEST(test_queue8_iterate_next); RUN_TEST(test_queue2_iterate_next_full); RUN_TEST(test_queue8_iterate_next_reset_on_change); test_print_result(); }