bag.c 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. /* bag.c -- ARMulator support code: ARM6 Instruction Emulator.
  2. Copyright (C) 1994 Advanced RISC Machines Ltd.
  3. This program is free software; you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation; either version 3 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with this program; if not, see <http://www.gnu.org/licenses/>. */
  13. /********************************************************************/
  14. /* bag.c: */
  15. /* Offers a data structure for storing and getting pairs of number. */
  16. /* The numbers are stored together, put one can be looked up by */
  17. /* quoting the other. If a new pair is entered and one of the */
  18. /* numbers is a repeat of a previous pair, then the previos pair */
  19. /* is deleted. */
  20. /********************************************************************/
  21. #include "bag.h"
  22. #include <stdlib.h>
  23. #define HASH_TABLE_SIZE 256
  24. #define hash(x) (((x)&0xff)^(((x)>>8)&0xff)^(((x)>>16)&0xff)^(((x)>>24)&0xff))
  25. typedef struct hashentry
  26. {
  27. struct hashentry *next;
  28. int first;
  29. int second;
  30. }
  31. Hashentry;
  32. Hashentry *lookupbyfirst[HASH_TABLE_SIZE];
  33. Hashentry *lookupbysecond[HASH_TABLE_SIZE];
  34. static void
  35. addtolist (Hashentry ** add, long first, long second)
  36. {
  37. while (*add)
  38. add = &((*add)->next);
  39. /* Malloc will never fail? :o( */
  40. (*add) = (Hashentry *) malloc (sizeof (Hashentry));
  41. (*add)->next = (Hashentry *) 0;
  42. (*add)->first = first;
  43. (*add)->second = second;
  44. }
  45. static void
  46. killwholelist (Hashentry * p)
  47. {
  48. Hashentry *q;
  49. while (p)
  50. {
  51. q = p;
  52. p = p->next;
  53. free (q);
  54. }
  55. }
  56. static void
  57. removefromlist (Hashentry ** p, long first)
  58. {
  59. Hashentry *q;
  60. while (*p)
  61. {
  62. if ((*p)->first == first)
  63. {
  64. q = (*p)->next;
  65. free (*p);
  66. *p = q;
  67. return;
  68. }
  69. p = &((*p)->next);
  70. }
  71. }
  72. void
  73. BAG_putpair (long first, long second)
  74. {
  75. long junk;
  76. if (BAG_getfirst (&junk, second) != NO_SUCH_PAIR)
  77. BAG_killpair_bysecond (second);
  78. addtolist (&lookupbyfirst[hash (first)], first, second);
  79. addtolist (&lookupbysecond[hash (second)], first, second);
  80. }
  81. Bag_error
  82. BAG_getfirst (long *first, long second)
  83. {
  84. Hashentry *look;
  85. look = lookupbysecond[hash (second)];
  86. while (look)
  87. if (look->second == second)
  88. {
  89. *first = look->first;
  90. return NO_ERROR;
  91. }
  92. return NO_SUCH_PAIR;
  93. }
  94. Bag_error
  95. BAG_getsecond (long first, long *second)
  96. {
  97. Hashentry *look;
  98. look = lookupbyfirst[hash (first)];
  99. while (look)
  100. {
  101. if (look->first == first)
  102. {
  103. *second = look->second;
  104. return NO_ERROR;
  105. }
  106. look = look->next;
  107. }
  108. return NO_SUCH_PAIR;
  109. }
  110. Bag_error
  111. BAG_killpair_byfirst (long first)
  112. {
  113. long second;
  114. if (BAG_getsecond (first, &second) == NO_SUCH_PAIR)
  115. return NO_SUCH_PAIR;
  116. removefromlist (&lookupbyfirst[hash (first)], first);
  117. removefromlist (&lookupbysecond[hash (second)], first);
  118. return NO_ERROR;
  119. }
  120. Bag_error
  121. BAG_killpair_bysecond (long second)
  122. {
  123. long first;
  124. if (BAG_getfirst (&first, second) == NO_SUCH_PAIR)
  125. return NO_SUCH_PAIR;
  126. removefromlist (&lookupbyfirst[hash (first)], first);
  127. removefromlist (&lookupbysecond[hash (second)], first);
  128. return NO_ERROR;
  129. }
  130. void
  131. BAG_newbag (void)
  132. {
  133. int i;
  134. for (i = 0; i < 256; i++)
  135. {
  136. killwholelist (lookupbyfirst[i]);
  137. killwholelist (lookupbysecond[i]);
  138. lookupbyfirst[i] = lookupbysecond[i] = (Hashentry *) 0;
  139. }
  140. }