global_id.cc 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. /********************************************************************** <BR>
  2. This file is part of Crack dot Com's free source code release of
  3. Golgotha. <a href="http://www.crack.com/golgotha_release"> <BR> for
  4. information about compiling & licensing issues visit this URL</a>
  5. <PRE> If that doesn't help, contact Jonathan Clark at
  6. golgotha_source@usa.net (Subject should have "GOLG" in it)
  7. ***********************************************************************/
  8. #include "global_id.hh"
  9. #include "error/error.hh"
  10. #include "memory/malloc.hh"
  11. #include "saver.hh"
  12. #include "g1_object.hh"
  13. #include <string.h>
  14. sw32 g1_cur_num_map_objects;
  15. #define PREASSIGN_SIG ((g1_object_class *)0xbadfeed)
  16. g1_global_id_manager_class::g1_global_id_manager_class()
  17. {
  18. init();
  19. claim_freespace();
  20. }
  21. i4_bool g1_global_id_manager_class::preassigned(w32 id) const
  22. {
  23. return obj[id&ID_MASK]==PREASSIGN_SIG;
  24. }
  25. g1_global_id_reset_notifier *g1_global_id_reset_notifier::first=0;
  26. g1_global_id_reset_notifier::g1_global_id_reset_notifier()
  27. {
  28. next=first;
  29. first=this;
  30. }
  31. g1_global_id_reset_notifier::~g1_global_id_reset_notifier()
  32. {
  33. if (first==this)
  34. first=first->next;
  35. else
  36. {
  37. g1_global_id_reset_notifier *p;
  38. for (p=first; p->next!=this; p=p->next);
  39. p->next=next;
  40. }
  41. }
  42. void g1_global_id_manager_class::init()
  43. {
  44. // initialize ids
  45. for (w32 i=0; i<G1_MAX_OBJECTS; i++)
  46. obj_id[i] = ID_INCREMENT+i;
  47. memset(obj, 0, sizeof(obj));
  48. first_free=G1_MAX_OBJECTS;
  49. g1_cur_num_map_objects=0;
  50. for (g1_global_id_reset_notifier *p=g1_global_id_reset_notifier::first;
  51. p; p=p->next)
  52. p->reset();
  53. }
  54. void g1_global_id_manager_class::free_objects()
  55. {
  56. w32 i = first_free,j=0;
  57. while (i<G1_MAX_OBJECTS)
  58. {
  59. j = *((w32*)(&obj[i])); // next one
  60. obj[i] = 0;
  61. i = j;
  62. }
  63. for (i=0; i<G1_MAX_OBJECTS; i++)
  64. if (obj[i])
  65. {
  66. delete obj[i];
  67. obj[i]=0;
  68. }
  69. first_free=G1_MAX_OBJECTS;
  70. }
  71. void g1_global_id_manager_class::claim_freespace()
  72. {
  73. int i=G1_MAX_OBJECTS;
  74. while (i>0)
  75. {
  76. i--;
  77. if (!obj[i])
  78. {
  79. *((w32*)(&obj[i]))=first_free;
  80. first_free=i;
  81. }
  82. }
  83. }
  84. void g1_global_id_manager_class::assign(w32 id, g1_object_class *for_who)
  85. {
  86. w32 index = id&ID_MASK;
  87. if (obj_id[index]!=id)
  88. i4_warning("assigning a possibly invalid id!");
  89. if (obj[index]!=PREASSIGN_SIG)
  90. i4_warning("assigning a previously assigned id!");
  91. obj[index] = for_who;
  92. //return i4_T;
  93. }
  94. w32 g1_global_id_manager_class::alloc(g1_object_class *for_who)
  95. {
  96. if (first_free>=G1_MAX_OBJECTS)
  97. {
  98. claim_freespace();
  99. if (first_free>=G1_MAX_OBJECTS)
  100. i4_error("alloc object id : too many objects");
  101. }
  102. w32 id=first_free;
  103. first_free = *((w32*)(&obj[id]));
  104. obj[id] = (for_who) ? for_who : PREASSIGN_SIG;
  105. g1_cur_num_map_objects++;
  106. return obj_id[id];
  107. }
  108. void g1_global_id_manager_class::free(w32 id)
  109. {
  110. if (!check_id(id))
  111. {
  112. i4_warning("free object id : bad id");
  113. return;
  114. }
  115. id &= ID_MASK;
  116. #if 1
  117. // forces use of all ids first
  118. obj[id]=0;
  119. #else
  120. *((w32*)(&obj[id]))=first_free;
  121. first_free=id;
  122. #endif
  123. obj_id[id] += ID_INCREMENT;
  124. g1_cur_num_map_objects--;
  125. }
  126. g1_global_id_manager_class::remapper::remapper(g1_global_id_manager_class *gid) : gid(gid)
  127. {
  128. map = (w32 *)i4_malloc(G1_MAX_OBJECTS*sizeof(*map), "global_id_remapping");
  129. }
  130. g1_global_id_manager_class::remapper::~remapper()
  131. {
  132. i4_free(map);
  133. }
  134. g1_global_id_manager_class g1_global_id;
  135. void g1_id_ref::save(g1_saver_class *fp)
  136. {
  137. fp->write_global_id(id);
  138. }
  139. void g1_id_ref::load(g1_loader_class *fp)
  140. {
  141. id=fp->read_global_id();
  142. }
  143. g1_id_ref::g1_id_ref(g1_object_class *o)
  144. {
  145. id=o->global_id;
  146. }
  147. g1_id_ref& g1_id_ref::operator=(g1_object_class *o)
  148. {
  149. if (o)
  150. id=o->global_id;
  151. else
  152. id=g1_global_id.invalid_id();
  153. return *this;
  154. }
  155. void g1_global_id_manager_class::debug(w32 flag_pass)
  156. {
  157. w32 invalid[(G1_MAX_OBJECTS+31)/32];
  158. w32 num_free=0;
  159. memset(invalid,0,sizeof(invalid));
  160. w32 i = first_free;
  161. while (i<G1_MAX_OBJECTS)
  162. {
  163. invalid[i>>5] |= (1<< (i&31)); // mark bit
  164. i = *((w32*)(&obj[i])); // next one
  165. num_free++; // count
  166. }
  167. for (i=0; i<G1_MAX_OBJECTS; i++)
  168. if ((invalid[i>>5] & (1<<(i&31)))==0)
  169. if (!obj[i])
  170. {
  171. invalid[i>>5] |= (1<< (i&31));
  172. num_free++;
  173. }
  174. else
  175. if (obj[i]->get_flag(flag_pass))
  176. i4_debug->printf("%d: [%s] team:%d [%c%c%c%c]\n",
  177. i,
  178. obj[i]->name(),
  179. obj[i]->player_num,
  180. obj[i]->get_flag(g1_object_class::MAP_OCCUPIED)?'M':'m',
  181. obj[i]->get_flag(g1_object_class::THINKING)?'T':'t',
  182. obj[i]->get_flag(g1_object_class::SCRATCH_BIT)?'S':'s',
  183. obj[i]->get_flag(g1_object_class::DELETED)?'D':'d'
  184. );
  185. i4_debug->printf("Valid Objects: %d Free Spaces: %d\n", G1_MAX_OBJECTS-num_free, num_free);
  186. }
  187. void g1_list_objects(w32 flag_pass=0xffffffff)
  188. {
  189. g1_global_id.debug(flag_pass);
  190. }