strlist.c 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. /*
  2. * Part of Very Secure FTPd
  3. * Licence: GPL v2
  4. * Author: Chris Evans
  5. * strlist.c
  6. */
  7. /* Anti-lamer measures deployed, sir! */
  8. #define PRIVATE_HANDS_OFF_alloc_len alloc_len
  9. #define PRIVATE_HANDS_OFF_list_len list_len
  10. #define PRIVATE_HANDS_OFF_p_nodes p_nodes
  11. #include "strlist.h"
  12. #include "str.h"
  13. #include "utility.h"
  14. #include "sysutil.h"
  15. struct mystr_list_node
  16. {
  17. struct mystr str;
  18. struct mystr sort_key_str;
  19. };
  20. /* File locals */
  21. static const unsigned int kMaxStrlist = 10 * 1000 * 1000;
  22. static struct mystr s_null_str;
  23. static int sort_compare_func(const void* p1, const void* p2);
  24. static int sort_compare_func_reverse(const void* p1, const void* p2);
  25. static int sort_compare_common(const void* p1, const void* p2, int reverse);
  26. void
  27. str_list_free(struct mystr_list* p_list)
  28. {
  29. unsigned int i;
  30. for (i=0; i < p_list->list_len; ++i)
  31. {
  32. str_free(&p_list->p_nodes[i].str);
  33. str_free(&p_list->p_nodes[i].sort_key_str);
  34. }
  35. p_list->list_len = 0;
  36. p_list->alloc_len = 0;
  37. if (p_list->p_nodes)
  38. {
  39. vsf_sysutil_free(p_list->p_nodes);
  40. p_list->p_nodes = 0;
  41. }
  42. }
  43. unsigned int
  44. str_list_get_length(const struct mystr_list* p_list)
  45. {
  46. return p_list->list_len;
  47. }
  48. int
  49. str_list_contains_str(const struct mystr_list* p_list,
  50. const struct mystr* p_str)
  51. {
  52. unsigned int i;
  53. for (i=0; i < p_list->list_len; ++i)
  54. {
  55. if (str_equal(p_str, &p_list->p_nodes[i].str))
  56. {
  57. return 1;
  58. }
  59. }
  60. return 0;
  61. }
  62. void
  63. str_list_add(struct mystr_list* p_list, const struct mystr* p_str,
  64. const struct mystr* p_sort_key_str)
  65. {
  66. struct mystr_list_node* p_node;
  67. /* Expand the node allocation if we have to */
  68. if (p_list->list_len == p_list->alloc_len)
  69. {
  70. if (p_list->alloc_len == 0)
  71. {
  72. p_list->alloc_len = 32;
  73. p_list->p_nodes = vsf_sysutil_malloc(
  74. p_list->alloc_len * (unsigned int) sizeof(struct mystr_list_node));
  75. }
  76. else
  77. {
  78. p_list->alloc_len *= 2;
  79. if (p_list->alloc_len > kMaxStrlist)
  80. {
  81. die("excessive strlist");
  82. }
  83. p_list->p_nodes = vsf_sysutil_realloc(
  84. p_list->p_nodes,
  85. p_list->alloc_len * (unsigned int) sizeof(struct mystr_list_node));
  86. }
  87. }
  88. p_node = &p_list->p_nodes[p_list->list_len];
  89. p_node->str = s_null_str;
  90. p_node->sort_key_str = s_null_str;
  91. str_copy(&p_node->str, p_str);
  92. if (p_sort_key_str)
  93. {
  94. str_copy(&p_node->sort_key_str, p_sort_key_str);
  95. }
  96. p_list->list_len++;
  97. }
  98. void
  99. str_list_sort(struct mystr_list* p_list, int reverse)
  100. {
  101. if (!reverse)
  102. {
  103. vsf_sysutil_qsort(p_list->p_nodes, p_list->list_len,
  104. sizeof(struct mystr_list_node), sort_compare_func);
  105. }
  106. else
  107. {
  108. vsf_sysutil_qsort(p_list->p_nodes, p_list->list_len,
  109. sizeof(struct mystr_list_node),
  110. sort_compare_func_reverse);
  111. }
  112. }
  113. static int
  114. sort_compare_func(const void* p1, const void* p2)
  115. {
  116. return sort_compare_common(p1, p2, 0);
  117. }
  118. static int
  119. sort_compare_func_reverse(const void* p1, const void* p2)
  120. {
  121. return sort_compare_common(p1, p2, 1);
  122. }
  123. static int
  124. sort_compare_common(const void* p1, const void* p2, int reverse)
  125. {
  126. const struct mystr* p_cmp1;
  127. const struct mystr* p_cmp2;
  128. const struct mystr_list_node* p_node1 = (const struct mystr_list_node*) p1;
  129. const struct mystr_list_node* p_node2 = (const struct mystr_list_node*) p2;
  130. if (!str_isempty(&p_node1->sort_key_str))
  131. {
  132. p_cmp1 = &p_node1->sort_key_str;
  133. }
  134. else
  135. {
  136. p_cmp1 = &p_node1->str;
  137. }
  138. if (!str_isempty(&p_node2->sort_key_str))
  139. {
  140. p_cmp2 = &p_node2->sort_key_str;
  141. }
  142. else
  143. {
  144. p_cmp2 = &p_node2->str;
  145. }
  146. if (reverse)
  147. {
  148. return str_strcmp(p_cmp2, p_cmp1);
  149. }
  150. else
  151. {
  152. return str_strcmp(p_cmp1, p_cmp2);
  153. }
  154. }
  155. const struct mystr*
  156. str_list_get_pstr(const struct mystr_list* p_list, unsigned int indexx)
  157. {
  158. if (indexx >= p_list->list_len)
  159. {
  160. bug("indexx out of range in str_list_get_str");
  161. }
  162. return &p_list->p_nodes[indexx].str;
  163. }