specs.c 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008
  1. #include "image.hpp"
  2. #include "palette.hpp"
  3. #include "specs.hpp"
  4. #include <stdio.h>
  5. #include <stdlib.h>
  6. #include <ctype.h>
  7. #include <fcntl.h>
  8. #include "system.h"
  9. #include "jmalloc.hpp"
  10. #include <math.h>
  11. #include "dprint.hpp"
  12. #ifndef __POWERPC__
  13. #include <sys/types.h>
  14. #include <sys/stat.h>
  15. #endif
  16. #ifdef __POWERPC__
  17. extern char *macify_name(char *s);
  18. #endif
  19. char *spec_types[]={"Invalid type", // 0
  20. "Color table", // 1
  21. "Palette", // 2
  22. "Invalid Type", // 3
  23. "Image", // 4
  24. "Fore Tile",
  25. "Back Tile",
  26. "Character",
  27. "8 Morph",
  28. "16 Morph",
  29. "Grue objs",
  30. "Extern WAV",
  31. "DMX MUS",
  32. "Patched morph",
  33. "Normal file",
  34. "Compress1 file",
  35. "Vector Image",
  36. "Light list",
  37. "Grue fgmap",
  38. "Grue bgmap",
  39. "Data array",
  40. "Character2",
  41. "Particle",
  42. "Extern lcache"
  43. };
  44. int total_files_open=0;
  45. char spec_main_file[100];
  46. static char *spec_prefix=NULL;
  47. static char *save_spec_prefix=NULL;
  48. static jFILE spec_main_jfile((FILE*)0);
  49. static int spec_main_fd = -1;
  50. static long spec_main_offset = -1;
  51. static spec_directory spec_main_sd;
  52. static int fast_load_fd = -1;
  53. static int fast_load_mode = 0;
  54. void set_filename_prefix(char *prefix)
  55. {
  56. if (spec_prefix) { jfree(spec_prefix); }
  57. if (prefix)
  58. {
  59. spec_prefix=strcpy((char *)jmalloc(strlen(prefix)+2,"prefix_name"),prefix);
  60. int len=strlen(prefix);
  61. if (prefix[len-1]!='\\' && prefix[len-1]!='/')
  62. {
  63. spec_prefix[len]='/';
  64. spec_prefix[len+1]=0;
  65. }
  66. }
  67. else spec_prefix=NULL;
  68. }
  69. char *get_filename_prefix()
  70. {
  71. return spec_prefix;
  72. }
  73. void set_save_filename_prefix(char *save_prefix)
  74. {
  75. if (save_spec_prefix) { jfree(save_spec_prefix); }
  76. if (save_prefix)
  77. {
  78. save_spec_prefix=strcpy((char *)jmalloc(strlen(save_prefix)+1,"prefix_name"),save_prefix);
  79. int len=strlen(save_prefix);
  80. if (save_prefix[len-1]!='\\' && save_prefix[len-1]!='/')
  81. {
  82. save_spec_prefix[len]='/';
  83. save_spec_prefix[len+1]=0;
  84. }
  85. }
  86. else save_spec_prefix=NULL;
  87. }
  88. char *get_save_filename_prefix()
  89. {
  90. return save_spec_prefix;
  91. }
  92. int search_order=SPEC_SEARCH_OUTSIDE_INSIDE;
  93. static void (*no_space_handle_fun)()=NULL;
  94. void set_no_space_handler(void (*handle_fun)())
  95. {
  96. no_space_handle_fun=handle_fun;
  97. }
  98. bFILE::bFILE()
  99. {
  100. rbuf_size=8192;
  101. rbuf=(unsigned char *)jmalloc(rbuf_size,"File read buffer");
  102. rbuf_start=rbuf_end=0;
  103. wbuf_size=8192;
  104. wbuf=(unsigned char *)jmalloc(wbuf_size,"File write buffer");
  105. wbuf_end=0;
  106. }
  107. bFILE::~bFILE()
  108. {
  109. if (rbuf) jfree(rbuf);
  110. flush_writes();
  111. if (wbuf) jfree(wbuf);
  112. }
  113. int bFILE::flush_writes()
  114. {
  115. if (wbuf_end!=0)
  116. {
  117. long ret=unbuffered_write(wbuf,wbuf_end);
  118. if (ret!=wbuf_end && no_space_handle_fun)
  119. no_space_handle_fun();
  120. wbuf_end=0;
  121. return ret;
  122. }
  123. return 0;
  124. }
  125. int bFILE::read(void *buf, size_t count) // returns number of bytes read, calls unbuffer_read
  126. {
  127. if (!allow_read_buffering())
  128. return unbuffered_read(buf,count);
  129. int total_read=0,error=0;
  130. if (!count) return 0;
  131. while (count && !error)
  132. {
  133. if (rbuf_start<rbuf_end)
  134. {
  135. int avail_size=rbuf_end-rbuf_start;
  136. int copy_size=avail_size>count ? count : avail_size;
  137. memcpy(buf,rbuf+rbuf_start,copy_size);
  138. buf=(void *)(((unsigned char *)buf)+copy_size);
  139. rbuf_start+=copy_size;
  140. if (rbuf_start>=rbuf_end)
  141. {
  142. if (rbuf_end!=rbuf_size) // buffer wasn't full before so there is no way we can complete read
  143. error=1;
  144. rbuf_start=rbuf_end=0;
  145. }
  146. total_read+=copy_size;
  147. count-=copy_size;
  148. } else
  149. {
  150. rbuf_end=unbuffered_read(rbuf,rbuf_size);
  151. if (rbuf_end==0) error=1;
  152. rbuf_start=0;
  153. }
  154. }
  155. return total_read;
  156. }
  157. int bFILE::write(void *buf, size_t count) // returns number of bytes written
  158. {
  159. if (allow_write_buffering())
  160. {
  161. int total_written=0;
  162. while (count)
  163. {
  164. int copy_size=wbuf_end+count<=wbuf_size ? count : wbuf_size-wbuf_end;
  165. memcpy(wbuf+wbuf_end,buf,copy_size);
  166. wbuf_end+=copy_size;
  167. count-=copy_size;
  168. buf=(void *)(((char *)buf)+copy_size);
  169. if (wbuf_end==wbuf_size)
  170. if (flush_writes()!=wbuf_size)
  171. return total_written;
  172. total_written+=copy_size;
  173. }
  174. return total_written;
  175. } else
  176. {
  177. long ret=unbuffered_write(buf,count);
  178. if (ret!=count && no_space_handle_fun)
  179. no_space_handle_fun();
  180. }
  181. return 0;
  182. }
  183. int bFILE::seek(long offset, int whence) // whence=SEEK_SET, SEEK_CUR, SEEK_END, ret=0=success
  184. {
  185. // rbuf_start=rbuf_end=0;
  186. // unbuffered_seek(offset,SEEK_SET);
  187. long realpos=unbuffered_tell();
  188. long curpos=realpos-rbuf_end+rbuf_start;
  189. if (whence==SEEK_CUR) offset+=curpos;
  190. else if (whence==SEEK_END) offset=file_size()-offset;
  191. if (offset<realpos-rbuf_end || offset>=realpos)
  192. {
  193. rbuf_start=rbuf_end=0;
  194. unbuffered_seek(offset,SEEK_SET);
  195. } else
  196. rbuf_start=rbuf_end-(realpos-offset);
  197. return 1;
  198. }
  199. int bFILE::tell()
  200. {
  201. return unbuffered_tell()-rbuf_end+rbuf_start+
  202. wbuf_end; // if this a write file, add on how much we've written
  203. }
  204. int bFILE::allow_read_buffering() { return 1; }
  205. int bFILE::allow_write_buffering() { return 1; }
  206. void bFILE::set_read_buffer_size(long size)
  207. {
  208. unbuffered_seek(tell(),SEEK_SET);
  209. rbuf_start=rbuf_end=0;
  210. if (rbuf)
  211. jfree(rbuf);
  212. rbuf_size=size;
  213. rbuf=(unsigned char *)jmalloc(rbuf_size,"File buffer");
  214. }
  215. void set_spec_main_file(char *filename, int Search_order)
  216. {
  217. dprintf("Specs : main file set to %s\n",filename);
  218. strcpy(spec_main_file,filename);
  219. search_order=Search_order;
  220. #ifdef __POWERPC__
  221. spec_main_jfile.open_external(filename,"rb",O_BINARY|O_RDONLY);
  222. spec_main_fd = spec_main_jfile.get_fd();
  223. spec_main_sd.startup(&spec_main_jfile);
  224. #else
  225. spec_main_jfile.open_external(filename,"rb",O_RDONLY);
  226. spec_main_fd = spec_main_jfile.get_fd();
  227. spec_main_sd.startup(&spec_main_jfile);
  228. #endif
  229. }
  230. void fast_load_start_recording(char *filename)
  231. {
  232. #ifdef __POWERPC__
  233. fast_load_fd = ::open(macify_name(filename),O_BINARY|O_CREAT|O_RDWR);
  234. #else
  235. fast_load_fd = ::open(filename,O_CREAT|O_RDWR,S_IRWXU | S_IRWXG | S_IRWXO);
  236. #endif
  237. fast_load_mode = 1;
  238. }
  239. void fast_load_stop_recording()
  240. {
  241. fast_load_mode = 0;
  242. }
  243. void fast_load_start_reloading(char *filename)
  244. {
  245. #ifdef __POWERPC__
  246. fast_load_fd = ::open(macify_name(filename),O_BINARY|O_RDONLY);
  247. #else
  248. fast_load_fd = ::open(filename,O_RDONLY);
  249. #endif
  250. fast_load_mode = 2;
  251. }
  252. void fast_load_stop_reloading()
  253. {
  254. fast_load_mode = 0;
  255. }
  256. jFILE::jFILE(FILE *file_pointer) // assumes fp is at begining of file
  257. {
  258. access=0;
  259. fd=-1;
  260. file_length=0;
  261. start_offset=0;
  262. flags=JFILE_CLONED;
  263. }
  264. void jFILE::open_external(char *filename, char *mode, int flags)
  265. {
  266. int skip_size=0;
  267. char tmp_name[200];
  268. if (spec_prefix && filename[0] != '/')
  269. sprintf(tmp_name,"%s%s",spec_prefix,filename);
  270. else strcpy(tmp_name,filename);
  271. // int old_mask=umask(S_IRWXU | S_IRWXG | S_IRWXO);
  272. #ifdef __WATCOMC__
  273. flags|=O_BINARY;
  274. #endif
  275. if (flags&O_WRONLY)
  276. {
  277. if ((flags&O_APPEND)==0)
  278. {
  279. skip_size=1;
  280. int errval = unlink(tmp_name);
  281. }
  282. flags-=O_WRONLY;
  283. flags|=O_CREAT|O_RDWR;
  284. #ifdef __POWERPC__
  285. fd=open(macify_name(tmp_name),flags);
  286. #else
  287. fd=open(tmp_name,flags,S_IRWXU | S_IRWXG | S_IRWXO);
  288. #endif
  289. } else
  290. #ifdef __POWERPC__
  291. fd=open(macify_name(tmp_name),flags);
  292. #else
  293. fd=open(tmp_name,flags);
  294. #endif
  295. // umask(old_mask);
  296. if (fd>=0 && !skip_size)
  297. {
  298. file_length=lseek(fd,0,SEEK_END);
  299. if ((flags&O_APPEND)==0)
  300. lseek(fd,0,SEEK_SET);
  301. else
  302. current_offset = file_length;
  303. start_offset=0;
  304. } else
  305. {
  306. file_length=0;
  307. start_offset=0;
  308. }
  309. }
  310. class null_file : public bFILE // this file type will use virtual opens inside of a spe
  311. {
  312. public :
  313. virtual int open_failure() { return 1; }
  314. virtual int unbuffered_read(void *buf, size_t count) { return 0; }
  315. virtual int unbuffered_write(void *buf, size_t count) { return 0; }
  316. virtual int unbuffered_seek(long offset, int whence) { return 0; }
  317. virtual int unbuffered_tell() { return 0; }
  318. virtual int file_size() { return 0; }
  319. virtual ~null_file() { ; }
  320. } ;
  321. static bFILE *(*open_file_fun)(char *,char *)=NULL;
  322. int (*verify_file_fun)(char *,char *)=NULL;
  323. void set_file_opener(bFILE *(*open_fun)(char *, char *))
  324. {
  325. open_file_fun=open_fun;
  326. }
  327. bFILE *open_file(char *filename, char *mode)
  328. {
  329. if (!verify_file_fun || verify_file_fun(filename,mode))
  330. {
  331. if (open_file_fun)
  332. return open_file_fun(filename,mode);
  333. else return new jFILE(filename,mode);
  334. } else return new null_file;
  335. }
  336. void jFILE::open_internal(char *filename, char *mode, int flags)
  337. {
  338. int wr=0;
  339. for (char *s=mode;*s;s++)
  340. if (toupper(*s)=='A' || toupper(*s)=='W')
  341. wr=1;
  342. if (wr)
  343. fd=-1; // only allow extern file openings for writing
  344. else
  345. {
  346. fd = spec_main_fd;
  347. if (fd>=0) // if we were able to open the main file, see if it's in there
  348. {
  349. start_offset=0;
  350. spec_entry *se=spec_main_sd.find(filename);
  351. if (se)
  352. {
  353. start_offset=se->offset;
  354. current_offset = 0;
  355. file_length=se->size;
  356. rbuf_start=rbuf_end=0;
  357. } else
  358. {
  359. close(fd);
  360. fd=-1;
  361. }
  362. }
  363. }
  364. }
  365. jFILE::jFILE(char *filename, char *access_string) // same as fopen parameters
  366. {
  367. flags=access=0;
  368. char *s=access_string;
  369. for (;*s;s++)
  370. if (toupper(*s)=='R') access=O_RDONLY;
  371. for (s=access_string;*s;s++)
  372. if (toupper(*s)=='W')
  373. if (access)
  374. access=O_RDWR;
  375. else access=O_WRONLY;
  376. for (s=access_string;*s;s++)
  377. if (toupper(*s)=='A')
  378. access|=O_APPEND|O_WRONLY;
  379. file_length=start_offset=-1;
  380. current_offset = 0;
  381. fd=-1;
  382. if (search_order==SPEC_SEARCH_OUTSIDE_INSIDE)
  383. open_external(filename,access_string,access);
  384. if (fd<0)
  385. open_internal(filename,access_string,access);
  386. if (fd<0 && search_order==SPEC_SEARCH_INSIDE_OUTSIDE)
  387. open_external(filename,access_string,access);
  388. total_files_open++;
  389. }
  390. jFILE::~jFILE()
  391. {
  392. flush_writes();
  393. if (fd>=0 && !(flags&JFILE_CLONED))
  394. {
  395. total_files_open--;
  396. if (fd != spec_main_fd)
  397. close(fd);
  398. }
  399. }
  400. int jFILE::unbuffered_tell()
  401. {
  402. // int ret = ::lseek(fd,0,SEEK_CUR) - start_offset;
  403. // if (ret != current_offset)
  404. // fprintf(stderr,"Bad tell %d\n",current_offset);
  405. return current_offset;
  406. }
  407. int jFILE::unbuffered_read(void *buf, size_t count)
  408. {
  409. long len;
  410. if (fd == spec_main_fd)
  411. {
  412. switch (fast_load_mode)
  413. {
  414. case 0:
  415. if (current_offset+start_offset != spec_main_offset)
  416. spec_main_offset = lseek(fd, start_offset+current_offset, SEEK_SET);
  417. len = ::read(fd,(char*)buf,count);
  418. break;
  419. case 1:
  420. if (current_offset+start_offset != spec_main_offset)
  421. spec_main_offset = lseek(fd, start_offset+current_offset, SEEK_SET);
  422. len = ::read(fd,(char*)buf,count);
  423. ::write(fast_load_fd,(char*)&len,sizeof(len));
  424. ::write(fast_load_fd,(char*)buf,count);
  425. break;
  426. case 2:
  427. ::read(fast_load_fd,(char*)&len,sizeof(len));
  428. len = ::read(fast_load_fd,(char*)buf,len);
  429. break;
  430. }
  431. spec_main_offset += len;
  432. }
  433. else
  434. {
  435. switch (fast_load_mode)
  436. {
  437. case 0:
  438. len = ::read(fd,(char*)buf,count);
  439. break;
  440. case 1:
  441. len = ::read(fd,(char*)buf,count);
  442. ::write(fast_load_fd,(char*)&len,sizeof(len));
  443. ::write(fast_load_fd,(char*)buf,count);
  444. break;
  445. case 2:
  446. ::read(fast_load_fd,(char*)&len,sizeof(len));
  447. len = ::read(fast_load_fd,(char*)buf,len);
  448. if (count != len)
  449. printf("short read! %d:%d\n",current_offset,len);
  450. break;
  451. }
  452. }
  453. current_offset += len;
  454. return len;
  455. }
  456. int jFILE::unbuffered_write(void *buf, size_t count)
  457. {
  458. long ret = ::write(fd,(char*)buf,count);
  459. current_offset += ret;
  460. return ret;
  461. }
  462. int jFILE::unbuffered_seek(long offset, int whence) // whence=SEEK_SET, SEEK_CUR, SEEK_END, ret=0=success
  463. {
  464. long ret;
  465. if (fast_load_mode == 2)
  466. {
  467. switch (whence)
  468. {
  469. case SEEK_SET :
  470. current_offset = start_offset+offset;
  471. break;
  472. case SEEK_END :
  473. current_offset = start_offset+file_length-offset;
  474. break;
  475. case SEEK_CUR :
  476. current_offset += offset;
  477. break;
  478. default:
  479. ret = -1;
  480. break;
  481. }
  482. return current_offset;
  483. }
  484. switch (whence)
  485. {
  486. case SEEK_SET :
  487. { ret = lseek(fd,start_offset+offset,SEEK_SET); } break;
  488. case SEEK_END :
  489. { ret = lseek(fd,start_offset+file_length-offset,SEEK_SET); } break;
  490. case SEEK_CUR :
  491. { ret = lseek(fd,offset,SEEK_CUR); } break;
  492. default:
  493. ret = -1;
  494. break;
  495. }
  496. if (ret>=0)
  497. {
  498. current_offset = ret - start_offset;
  499. if (spec_main_fd == fd)
  500. spec_main_offset = ret;
  501. return ret;
  502. }
  503. else
  504. return -1; // if a bad whence, then failure
  505. }
  506. unsigned char bFILE::read_byte()
  507. { unsigned char x;
  508. read(&x,1);
  509. return x;
  510. }
  511. unsigned short bFILE::read_short()
  512. {
  513. unsigned short x;
  514. read(&x,2);
  515. return int_to_local(x);
  516. }
  517. unsigned long bFILE::read_long()
  518. {
  519. unsigned long x;
  520. read(&x,4);
  521. return long_to_local(x);
  522. }
  523. void bFILE::write_byte(unsigned char x)
  524. {
  525. write(&x,1);
  526. }
  527. void bFILE::write_short(unsigned short x)
  528. {
  529. x=int_to_local(x);
  530. write(&x,2);
  531. }
  532. void bFILE::write_long(unsigned long x)
  533. {
  534. x=long_to_local(x);
  535. write(&x,4);
  536. }
  537. void bFILE::write_double(double x)
  538. {
  539. double a;
  540. write_long((long)(modf(x,&a)*(double)(1<<32-1)));
  541. write_long((long)a);
  542. }
  543. double bFILE::read_double()
  544. {
  545. long a,b;
  546. a=read_long();
  547. b=read_long();
  548. return (double)b+a/(double)(1<<32-1);
  549. }
  550. spec_directory::~spec_directory()
  551. {
  552. if (total)
  553. {
  554. jfree(data);
  555. jfree(entries);
  556. }
  557. }
  558. void spec_entry::print()
  559. {
  560. printf("%15s%25s%8ld%8ld\n",spec_types[type],name,size,offset);
  561. }
  562. void spec_directory::calc_offsets()
  563. {
  564. spec_entry **e;
  565. int i;
  566. long o=SPEC_SIG_SIZE+2;
  567. if (total)
  568. {
  569. for (i=0,e=entries;i<total;i++,e++) // calculate the size of directory info
  570. {
  571. o+=1+1+strlen((*e)->name)+1 +1 +8;
  572. }
  573. for (i=0,e=entries;i<total;i++,e++) // calculate offset for each entry
  574. {
  575. (*e)->offset=o;
  576. o+=(*e)->size;
  577. }
  578. }
  579. }
  580. spec_entry *spec_directory::find(char *name, int type)
  581. {
  582. int i;
  583. spec_entry **e;
  584. for (i=0,e=entries;i<total;i++,e++)
  585. if (!strcmp((*e)->name,name) && (*e)->type==type)
  586. return (*e);
  587. return NULL;
  588. }
  589. spec_entry *spec_directory::find(char *name)
  590. {
  591. int i;
  592. spec_entry **e;
  593. for (i=0,e=entries;i<total;i++,e++)
  594. if (!strcmp((*e)->name,name))
  595. return (*e);
  596. return NULL;
  597. }
  598. long spec_directory::find_number(char *name)
  599. {
  600. int i;
  601. spec_entry **e;
  602. for (i=0,e=entries;i<total;i++,e++)
  603. if (!strcmp((*e)->name,name))
  604. return i;
  605. return -1;
  606. }
  607. spec_entry *spec_directory::find(int type)
  608. {
  609. int i;
  610. spec_entry **e;
  611. for (i=0,e=entries;i<total;i++,e++)
  612. if ((*e)->type==type)
  613. return (*e);
  614. return NULL;
  615. }
  616. long spec_directory::type_total(int type)
  617. {
  618. int i,x=0;
  619. spec_entry **e;
  620. for (i=0,e=entries;i<total;i++,e++)
  621. if ((*e)->type==type) x++;
  622. return x;
  623. }
  624. long spec_directory::find_number(int type)
  625. {
  626. int i;
  627. spec_entry **e;
  628. for (i=0,e=entries;i<total;i++,e++)
  629. if ((*e)->type==type)
  630. return i;
  631. return -1;
  632. }
  633. void spec_directory::print()
  634. {
  635. spec_entry **se;
  636. int i;
  637. printf("[ Entry type ][ Entry name ][ Size ][ Offset ]\n");
  638. for (i=0,se=entries;i<total;i++,se++)
  639. (*se)->print();
  640. }
  641. void spec_directory::startup(bFILE *fp)
  642. {
  643. char buf[256];
  644. fp->read(buf,8);
  645. buf[9]=0;
  646. size=0;
  647. if (!strcmp(buf,SPEC_SIGNATURE))
  648. {
  649. total=fp->read_short();
  650. entries=(spec_entry **)jmalloc(sizeof(spec_entry *)*total,"spec_directory::entries");
  651. long start=fp->tell();
  652. int i;
  653. for (i=0;i<total;i++)
  654. {
  655. fp->read(buf,2);
  656. long entry_size=sizeof(spec_entry)+(unsigned char)buf[1];
  657. entry_size=(entry_size+3)&(~3);
  658. fp->read(buf,(unsigned char)buf[1]);
  659. fp->read(buf,9);
  660. size+=entry_size;
  661. }
  662. data=jmalloc(size,"spec_directory::data");
  663. char *dp=(char *)data;
  664. fp->seek(start,SEEK_SET);
  665. for (i=0;i<total;i++)
  666. {
  667. spec_entry *se=(spec_entry *)dp;
  668. entries[i]=se;
  669. unsigned char len,flags,type;
  670. fp->read(&type,1);
  671. fp->read(&len,1);
  672. se->type=type;
  673. se->name=dp+sizeof(spec_entry);
  674. fp->read(se->name,len);
  675. fp->read(&flags,1);
  676. se->size=fp->read_long();
  677. se->offset=fp->read_long();
  678. dp+=((sizeof(spec_entry)+len)+3)&(~3);
  679. }
  680. }
  681. else
  682. {
  683. total=0;
  684. data=NULL;
  685. entries=NULL;
  686. }
  687. }
  688. spec_directory::spec_directory(bFILE *fp)
  689. { startup(fp); }
  690. spec_directory::spec_directory(FILE *fp)
  691. {
  692. jFILE jfp(fp);
  693. startup(&jfp);
  694. }
  695. spec_directory::spec_directory()
  696. {
  697. size=0;
  698. total=0;
  699. data=NULL;
  700. entries=NULL;
  701. }
  702. /*
  703. spec_directory::spec_directory(char *filename)
  704. {
  705. jFILE *fp;
  706. if (filename)
  707. {
  708. fp=new jFILE(filename,"rb");
  709. if (!fp->open_failure())
  710. startup(fp);
  711. else
  712. {
  713. total=0;
  714. entries=NULL;
  715. }
  716. delete fp;
  717. } else printf("NULL filename to spec_directory::spec_directory\n");
  718. }*/
  719. int write_string(bFILE *fp, char *st)
  720. {
  721. unsigned char length=strlen(st)+1;
  722. if (fp->write(&length,1)!=1) return 0;
  723. if (fp->write(st,length)!=length) return 0;
  724. return 1;
  725. }
  726. long spec_directory::data_start_offset()
  727. {
  728. spec_entry *e;
  729. long o=0,i;
  730. for (i=0;i<total;i++)
  731. return entries[i]->offset;
  732. return SPEC_SIG_SIZE+2; // if no entries, then no data, but return where it would
  733. // start anyway
  734. }
  735. long spec_directory::data_end_offset()
  736. {
  737. spec_entry **e;
  738. long o=0,i;
  739. for (i=total-1,e=entries;i>=0;i--,e++)
  740. return (*e)->offset+(*e)->size;
  741. return SPEC_SIG_SIZE+2;
  742. }
  743. int spec_directory::write(bFILE *fp)
  744. {
  745. char sig[SPEC_SIG_SIZE];
  746. unsigned short tentries;
  747. unsigned char flags=0;
  748. unsigned long offset,data_size;
  749. spec_entry **e;
  750. strcpy(sig,SPEC_SIGNATURE);
  751. if (fp->write(sig,sizeof(sig))!=sizeof(sig)) return 0;
  752. fp->write_short(total);
  753. int i;
  754. for (i=0,e=entries;i<total;i++,e++)
  755. {
  756. if (fp->write(&(*e)->type,1)!=1) return 0;
  757. if (!write_string(fp,(*e)->name)) return 0;
  758. flags=0;
  759. if (fp->write(&flags,1)!=1) return 0;
  760. data_size=long_to_intel((*e)->size);
  761. if (fp->write((char *)&data_size,4)!=4) return 0;
  762. offset=long_to_intel((*e)->offset);
  763. if (fp->write((char *)&offset,4)!=4) return 0;
  764. }
  765. return 1;
  766. }
  767. jFILE *spec_directory::write(char *filename)
  768. {
  769. jFILE *fp;
  770. fp=new jFILE(filename,"wb");
  771. if (fp->open_failure()) { delete fp; return NULL; }
  772. if (!write(fp))
  773. {
  774. delete fp;
  775. return NULL;
  776. } else return fp;
  777. }
  778. unsigned short read_short(FILE *fp)
  779. {
  780. unsigned short x;
  781. fread(&x,1,2,fp);
  782. return int_to_local(x);
  783. }
  784. unsigned long read_long(FILE *fp)
  785. {
  786. unsigned long x;
  787. fread(&x,1,4,fp);
  788. return (long)long_to_local(x);
  789. }
  790. void write_short(FILE *fp, unsigned short x)
  791. {
  792. x=int_to_local(x);
  793. fwrite(&x,1,2,fp);
  794. }
  795. void write_long(FILE *fp, unsigned long x)
  796. {
  797. x=long_to_local(x);
  798. fwrite(&x,1,4,fp);
  799. }
  800. unsigned char read_byte(FILE *fp) { return fgetc(fp)&0xff; }
  801. void write_byte(FILE *fp, unsigned char x) { fputc(x,fp); }
  802. unsigned short read_other_long(FILE *fp)
  803. {
  804. unsigned long x;
  805. fread(&x,1,4,fp);
  806. return big_long_to_local(x);
  807. }
  808. unsigned long read_other_short(FILE *fp)
  809. {
  810. unsigned short x;
  811. fread(&x,1,2,fp);
  812. return big_short_to_local(x);
  813. }
  814. void write_other_short(FILE *fp, unsigned short x)
  815. {
  816. x=big_short_to_local(x);
  817. fwrite(&x,1,2,fp);
  818. }
  819. void write_other_long(FILE *fp, unsigned long x)
  820. {
  821. x=big_long_to_local(x);
  822. fwrite(&x,1,4,fp);
  823. }
  824. void spec_directory::remove(spec_entry *e)
  825. {
  826. int i;
  827. for (i=0;i<total && entries[i]!=e;i++); // find the entry in the array first
  828. if (entries[i]==e) // make sre it was found
  829. {
  830. delete e;
  831. total--;
  832. for (;i<total;i++) // compact the pointer array
  833. entries[i]=entries[i+1];
  834. entries=(spec_entry **)jrealloc(entries,sizeof(spec_entry *)*total,"spec_directory::entries");
  835. }
  836. else
  837. printf("Spec_directory::remove bad entry pointer\n");
  838. }
  839. void spec_directory::add_by_hand(spec_entry *e)
  840. {
  841. total++;
  842. entries=(spec_entry **)jrealloc(entries,sizeof(spec_entry *)*total,"spec_directory::entries");
  843. entries[total-1]=e;
  844. }
  845. void spec_directory::delete_entries() // if the directory was created by hand instead of by file
  846. {
  847. int i;
  848. for (i=0;i<total;i++)
  849. delete entries[i];
  850. if (total)
  851. jfree(entries);
  852. }
  853. void note_open_fd(int fd, char *str)
  854. {
  855. total_files_open++;
  856. }
  857. void note_close_fd(int fd)
  858. {
  859. total_files_open--;
  860. }
  861. void list_open_fds()
  862. {
  863. printf("Total open dos fds=%d\n",total_files_open);
  864. }