g1_file.cc 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  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 "error/alert.hh"
  9. #include "string/str_checksum.hh"
  10. #include "threads/threads.hh"
  11. #include "file/file.hh"
  12. #include "memory/array.hh"
  13. #include "file/file_man.hh"
  14. #include "main/main.hh"
  15. #include "checksum/checksum.hh"
  16. i4_critical_section_class cd_file_lock;
  17. static char *cd_image="golgotha.cd";
  18. struct g1_dir_entry
  19. {
  20. w32 offset;
  21. w32 checksum;
  22. w32 length;
  23. };
  24. int g1_dir_entry_compare(const g1_dir_entry *a, const g1_dir_entry *b)
  25. {
  26. if (a->checksum<b->checksum)
  27. return -1;
  28. else if (a->checksum>b->checksum)
  29. return 1;
  30. else return 0;
  31. }
  32. // this is the only file that is actually used by the game, the rest is just seeking
  33. i4_file_class *g1_single_file=0;
  34. void g1_cd_file_callback(w32 count, void *context);
  35. class g1_cd_file : public i4_file_class
  36. {
  37. public:
  38. w32 offset;
  39. w32 end_offset;
  40. w32 start_offset;
  41. i4_file_class::async_callback callback;
  42. void *context;
  43. i4_file_class *use_file;
  44. virtual w32 read (void *buffer, w32 size)
  45. {
  46. cd_file_lock.lock();
  47. if (use_file->tell()!=offset)
  48. use_file->seek(offset);
  49. if (offset+size>end_offset)
  50. size=end_offset-offset;
  51. w32 ret=use_file->read(buffer,size);
  52. offset+=ret;
  53. cd_file_lock.unlock();
  54. return ret;
  55. }
  56. virtual w32 write(const void *buffer, w32 size)
  57. {
  58. return 0;
  59. }
  60. virtual w32 seek (w32 _offset)
  61. {
  62. offset=start_offset+_offset;
  63. if (offset>end_offset)
  64. offset=end_offset;
  65. return offset;
  66. }
  67. virtual w32 size ()
  68. {
  69. return end_offset-start_offset;
  70. }
  71. virtual w32 tell ()
  72. {
  73. return offset-start_offset;
  74. }
  75. ~g1_cd_file()
  76. {
  77. if (use_file!=g1_single_file)
  78. delete use_file;
  79. }
  80. g1_cd_file(w32 start, w32 length, i4_file_class *fp)
  81. {
  82. use_file=fp;
  83. start_offset=start;
  84. offset=start_offset;
  85. end_offset=start_offset+length;
  86. }
  87. };
  88. void g1_cd_file_callback(w32 count, void *context)
  89. {
  90. cd_file_lock.unlock();
  91. ((g1_cd_file *)context)->callback(count, ((g1_cd_file *)context)->context);
  92. }
  93. class g1_file_manager_class : public i4_file_manager_class
  94. {
  95. i4_array<g1_dir_entry> entries;
  96. char current_cd_file[100];
  97. public:
  98. int g1_file_manager_class::find_checksum(w32 id)
  99. {
  100. g1_dir_entry find;
  101. find.checksum=id;
  102. return entries.binary_search(&find, g1_dir_entry_compare);
  103. }
  104. virtual i4_file_class *open(const i4_const_str &name, w32 flags)
  105. {
  106. if (flags & (I4_WRITE|I4_APPEND|I4_SUPPORT_ASYNC))
  107. return 0;
  108. char tmp[256];
  109. int k=0;
  110. // convert slashes to a commonf format
  111. for (i4_const_str::iterator i=name.begin(); i!=name.end(); )
  112. {
  113. int c=i.get().value();
  114. if (c=='\\')
  115. tmp[k++]='/';
  116. else
  117. tmp[k++]=c;
  118. ++i;
  119. }
  120. int handle=find_checksum(i4_check_sum32(tmp,k));
  121. if (handle>=0)
  122. {
  123. if (flags & I4_SUPPORT_ASYNC)
  124. {
  125. i4_file_class *fp=i4_open(current_cd_file, flags);
  126. return new g1_cd_file(entries[handle].offset, entries[handle].length, fp);
  127. }
  128. return new g1_cd_file(entries[handle].offset, entries[handle].length, g1_single_file);
  129. }
  130. else return 0;
  131. }
  132. void set_cd_file()
  133. {
  134. if (cd_image)
  135. {
  136. g1_single_file=i4_open(cd_image);
  137. if (!g1_single_file)
  138. i4_warning("could not open cd image file");
  139. else
  140. {
  141. strcpy(current_cd_file, cd_image);
  142. int tfiles = g1_single_file->read_32();
  143. for (int i=0; i<tfiles; i++)
  144. {
  145. g1_dir_entry *e=entries.add();
  146. e->offset=g1_single_file->read_32();
  147. e->checksum=g1_single_file->read_32();
  148. e->length=g1_single_file->read_32();
  149. }
  150. int netfirst=0;
  151. for (int j=1; j<i4_global_argc; j++)
  152. if (i4_global_argv[j]=="-netfirst")
  153. netfirst=1;
  154. if (netfirst)
  155. i4_add_file_manager(this, i4_F);
  156. else
  157. i4_add_file_manager(this, i4_T);
  158. }
  159. }
  160. }
  161. void init()
  162. {
  163. set_cd_file();
  164. }
  165. void uninit()
  166. {
  167. entries.uninit();
  168. if (g1_single_file)
  169. {
  170. i4_remove_file_manger(this);
  171. delete g1_single_file;
  172. }
  173. }
  174. g1_file_manager_class()
  175. : entries(0,512)
  176. {
  177. }
  178. } g1_file_manager_class_instance;
  179. void g1_set_cd_image(char *filename)
  180. {
  181. cd_image=filename;
  182. g1_file_manager_class_instance.set_cd_file();
  183. }