/* Copyright (c) 2013-2017, The Tor Project, Inc. */ /* See LICENSE for licensing information */ #define CIRCUITLIST_PRIVATE #define RELAY_PRIVATE #include "or.h" #include "circuitlist.h" #include "relay.h" #include "test.h" static void test_cq_manip(void *arg) { packed_cell_t *pc1=NULL, *pc2=NULL, *pc3=NULL, *pc4=NULL, *pc_tmp=NULL; cell_queue_t cq; cell_t cell; (void) arg; cell_queue_init(&cq); tt_int_op(cq.n, OP_EQ, 0); pc1 = packed_cell_new(); pc2 = packed_cell_new(); pc3 = packed_cell_new(); pc4 = packed_cell_new(); tt_assert(pc1 && pc2 && pc3 && pc4); tt_ptr_op(NULL, OP_EQ, cell_queue_pop(&cq)); /* Add and remove a singleton. */ cell_queue_append(&cq, pc1); tt_int_op(cq.n, OP_EQ, 1); tt_ptr_op(pc1, OP_EQ, cell_queue_pop(&cq)); tt_int_op(cq.n, OP_EQ, 0); /* Add and remove four items */ cell_queue_append(&cq, pc4); cell_queue_append(&cq, pc3); cell_queue_append(&cq, pc2); cell_queue_append(&cq, pc1); tt_int_op(cq.n, OP_EQ, 4); tt_ptr_op(pc4, OP_EQ, cell_queue_pop(&cq)); tt_ptr_op(pc3, OP_EQ, cell_queue_pop(&cq)); tt_ptr_op(pc2, OP_EQ, cell_queue_pop(&cq)); tt_ptr_op(pc1, OP_EQ, cell_queue_pop(&cq)); tt_int_op(cq.n, OP_EQ, 0); tt_ptr_op(NULL, OP_EQ, cell_queue_pop(&cq)); /* Try a packed copy (wide, then narrow, which is a bit of a cheat, since a * real cell queue has only one type.) */ memset(&cell, 0, sizeof(cell)); cell.circ_id = 0x12345678; cell.command = 10; strlcpy((char*)cell.payload, "Lorax ipsum gruvvulus thneed amet, snergelly " "once-ler lerkim, sed do barbaloot tempor gluppitus ut labore et " "truffula magna aliqua.", sizeof(cell.payload)); cell_queue_append_packed_copy(NULL /*circ*/, &cq, 0 /*exitward*/, &cell, 1 /*wide*/, 0 /*stats*/); cell.circ_id = 0x2013; cell_queue_append_packed_copy(NULL /*circ*/, &cq, 0 /*exitward*/, &cell, 0 /*wide*/, 0 /*stats*/); tt_int_op(cq.n, OP_EQ, 2); pc_tmp = cell_queue_pop(&cq); tt_int_op(cq.n, OP_EQ, 1); tt_ptr_op(pc_tmp, OP_NE, NULL); tt_mem_op(pc_tmp->body, OP_EQ, "\x12\x34\x56\x78\x0a", 5); tt_mem_op(pc_tmp->body+5, OP_EQ, cell.payload, sizeof(cell.payload)); packed_cell_free(pc_tmp); pc_tmp = cell_queue_pop(&cq); tt_int_op(cq.n, OP_EQ, 0); tt_ptr_op(pc_tmp, OP_NE, NULL); tt_mem_op(pc_tmp->body, OP_EQ, "\x20\x13\x0a", 3); tt_mem_op(pc_tmp->body+3, OP_EQ, cell.payload, sizeof(cell.payload)); packed_cell_free(pc_tmp); pc_tmp = NULL; tt_ptr_op(NULL, OP_EQ, cell_queue_pop(&cq)); /* Now make sure cell_queue_clear works. */ cell_queue_append(&cq, pc2); cell_queue_append(&cq, pc1); tt_int_op(cq.n, OP_EQ, 2); cell_queue_clear(&cq); pc2 = pc1 = NULL; /* prevent double-free */ tt_int_op(cq.n, OP_EQ, 0); done: packed_cell_free(pc1); packed_cell_free(pc2); packed_cell_free(pc3); packed_cell_free(pc4); packed_cell_free(pc_tmp); cell_queue_clear(&cq); } static void test_circuit_n_cells(void *arg) { packed_cell_t *pc1=NULL, *pc2=NULL, *pc3=NULL, *pc4=NULL, *pc5=NULL; origin_circuit_t *origin_c=NULL; or_circuit_t *or_c=NULL; (void)arg; pc1 = packed_cell_new(); pc2 = packed_cell_new(); pc3 = packed_cell_new(); pc4 = packed_cell_new(); pc5 = packed_cell_new(); tt_assert(pc1 && pc2 && pc3 && pc4 && pc5); or_c = or_circuit_new(0, NULL); origin_c = origin_circuit_new(); origin_c->base_.purpose = CIRCUIT_PURPOSE_C_GENERAL; tt_int_op(n_cells_in_circ_queues(TO_CIRCUIT(or_c)), OP_EQ, 0); cell_queue_append(&or_c->p_chan_cells, pc1); tt_int_op(n_cells_in_circ_queues(TO_CIRCUIT(or_c)), OP_EQ, 1); cell_queue_append(&or_c->base_.n_chan_cells, pc2); cell_queue_append(&or_c->base_.n_chan_cells, pc3); tt_int_op(n_cells_in_circ_queues(TO_CIRCUIT(or_c)), OP_EQ, 3); tt_int_op(n_cells_in_circ_queues(TO_CIRCUIT(origin_c)), OP_EQ, 0); cell_queue_append(&origin_c->base_.n_chan_cells, pc4); cell_queue_append(&origin_c->base_.n_chan_cells, pc5); tt_int_op(n_cells_in_circ_queues(TO_CIRCUIT(origin_c)), OP_EQ, 2); done: circuit_free_(TO_CIRCUIT(or_c)); circuit_free_(TO_CIRCUIT(origin_c)); } struct testcase_t cell_queue_tests[] = { { "basic", test_cq_manip, TT_FORK, NULL, NULL, }, { "circ_n_cells", test_circuit_n_cells, TT_FORK, NULL, NULL }, END_OF_TESTCASES };