summaryrefslogtreecommitdiff
path: root/src/test/test_cell_queue.c
blob: 8916b067546b2aa3bbe25cd44521a8a3ea102646 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
/* Copyright (c) 2013, 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;

  init_cell_pool();
  cell_queue_init(&cq);
  tt_int_op(cq.n, ==, 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, ==, cell_queue_pop(&cq));

  /* Add and remove a singleton. */
  cell_queue_append(&cq, pc1);
  tt_int_op(cq.n, ==, 1);
  tt_ptr_op(pc1, ==, cell_queue_pop(&cq));
  tt_int_op(cq.n, ==, 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, ==, 4);
  tt_ptr_op(pc4, ==, cell_queue_pop(&cq));
  tt_ptr_op(pc3, ==, cell_queue_pop(&cq));
  tt_ptr_op(pc2, ==, cell_queue_pop(&cq));
  tt_ptr_op(pc1, ==, cell_queue_pop(&cq));
  tt_int_op(cq.n, ==, 0);
  tt_ptr_op(NULL, ==, 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(&cq, &cell, 1 /*wide*/, 0 /*stats*/);
  cell.circ_id = 0x2013;
  cell_queue_append_packed_copy(&cq, &cell, 0 /*wide*/, 0 /*stats*/);
  tt_int_op(cq.n, ==, 2);

  pc_tmp = cell_queue_pop(&cq);
  tt_int_op(cq.n, ==, 1);
  tt_ptr_op(pc_tmp, !=, NULL);
  test_mem_op(pc_tmp->body, ==, "\x12\x34\x56\x78\x0a", 5);
  test_mem_op(pc_tmp->body+5, ==, cell.payload, sizeof(cell.payload));
  packed_cell_free(pc_tmp);

  pc_tmp = cell_queue_pop(&cq);
  tt_int_op(cq.n, ==, 0);
  tt_ptr_op(pc_tmp, !=, NULL);
  test_mem_op(pc_tmp->body, ==, "\x20\x13\x0a", 3);
  test_mem_op(pc_tmp->body+3, ==, cell.payload, sizeof(cell.payload));
  packed_cell_free(pc_tmp);
  pc_tmp = NULL;

  tt_ptr_op(NULL, ==, 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, ==, 2);
  cell_queue_clear(&cq);
  pc2 = pc1 = NULL; /* prevent double-free */
  tt_int_op(cq.n, ==, 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);
  free_cell_pool();
}

struct testcase_t cell_queue_tests[] = {
  { "basic", test_cq_manip, TT_FORK, NULL, NULL, },
  END_OF_TESTCASES
};