SN_SONIX.C 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507
  1. //**************************************************************************
  2. //**
  3. //** sn_sonix.c : Heretic 2 : Raven Software, Corp.
  4. //**
  5. //** $RCSfile: sn_sonix.c,v $
  6. //** $Revision: 1.17 $
  7. //** $Date: 95/10/05 18:25:44 $
  8. //** $Author: paul $
  9. //**
  10. //**************************************************************************
  11. // HEADER FILES ------------------------------------------------------------
  12. #include <string.h>
  13. #include "h2def.h"
  14. #include "soundst.h"
  15. // MACROS ------------------------------------------------------------------
  16. #define SS_MAX_SCRIPTS 64
  17. #define SS_TEMPBUFFER_SIZE 1024
  18. #define SS_SEQUENCE_NAME_LENGTH 32
  19. #define SS_SCRIPT_NAME "SNDSEQ"
  20. #define SS_STRING_PLAY "play"
  21. #define SS_STRING_PLAYUNTILDONE "playuntildone"
  22. #define SS_STRING_PLAYTIME "playtime"
  23. #define SS_STRING_PLAYREPEAT "playrepeat"
  24. #define SS_STRING_DELAY "delay"
  25. #define SS_STRING_DELAYRAND "delayrand"
  26. #define SS_STRING_VOLUME "volume"
  27. #define SS_STRING_END "end"
  28. #define SS_STRING_STOPSOUND "stopsound"
  29. // TYPES -------------------------------------------------------------------
  30. typedef enum
  31. {
  32. SS_CMD_NONE,
  33. SS_CMD_PLAY,
  34. SS_CMD_WAITUNTILDONE, // used by PLAYUNTILDONE
  35. SS_CMD_PLAYTIME,
  36. SS_CMD_PLAYREPEAT,
  37. SS_CMD_DELAY,
  38. SS_CMD_DELAYRAND,
  39. SS_CMD_VOLUME,
  40. SS_CMD_STOPSOUND,
  41. SS_CMD_END
  42. } sscmds_t;
  43. // EXTERNAL FUNCTION PROTOTYPES --------------------------------------------
  44. // PUBLIC FUNCTION PROTOTYPES ----------------------------------------------
  45. // PRIVATE FUNCTION PROTOTYPES ---------------------------------------------
  46. static void VerifySequencePtr(int *base, int *ptr);
  47. static int GetSoundOffset(char *name);
  48. // EXTERNAL DATA DECLARATIONS ----------------------------------------------
  49. extern sfxinfo_t S_sfx[];
  50. // PUBLIC DATA DEFINITIONS -------------------------------------------------
  51. // PRIVATE DATA DEFINITIONS ------------------------------------------------
  52. static struct
  53. {
  54. char name[SS_SEQUENCE_NAME_LENGTH];
  55. int scriptNum;
  56. int stopSound;
  57. } SequenceTranslate[SEQ_NUMSEQ] =
  58. {
  59. { "Platform", 0, 0 },
  60. { "Platform", 0, 0 }, // a 'heavy' platform is just a platform
  61. { "PlatformMetal", 0, 0 },
  62. { "Platform", 0, 0 }, // same with a 'creak' platform
  63. { "Silence", 0, 0 },
  64. { "Lava", 0, 0 },
  65. { "Water", 0, 0 },
  66. { "Ice", 0, 0 },
  67. { "Earth", 0, 0 },
  68. { "PlatformMetal2", 0, 0 },
  69. { "DoorNormal", 0, 0 },
  70. { "DoorHeavy", 0, 0 },
  71. { "DoorMetal", 0, 0 },
  72. { "DoorCreak", 0, 0 },
  73. { "Silence", 0, 0 },
  74. { "Lava", 0, 0 },
  75. { "Water", 0, 0},
  76. { "Ice", 0, 0 },
  77. { "Earth", 0, 0},
  78. { "DoorMetal2", 0, 0 },
  79. { "Wind", 0, 0 }
  80. };
  81. static int *SequenceData[SS_MAX_SCRIPTS];
  82. int ActiveSequences;
  83. seqnode_t *SequenceListHead;
  84. // CODE --------------------------------------------------------------------
  85. //==========================================================================
  86. //
  87. // VerifySequencePtr
  88. //
  89. // Verifies the integrity of the temporary ptr, and ensures that the ptr
  90. // isn't exceeding the size of the temporary buffer
  91. //==========================================================================
  92. static void VerifySequencePtr(int *base, int *ptr)
  93. {
  94. if(ptr-base > SS_TEMPBUFFER_SIZE)
  95. {
  96. I_Error("VerifySequencePtr: tempPtr >= %d\n", SS_TEMPBUFFER_SIZE);
  97. }
  98. }
  99. //==========================================================================
  100. //
  101. // GetSoundOffset
  102. //
  103. //==========================================================================
  104. static int GetSoundOffset(char *name)
  105. {
  106. int i;
  107. for(i = 0; i < NUMSFX; i++)
  108. {
  109. if(!strcasecmp(name, S_sfx[i].tagName))
  110. {
  111. return i;
  112. }
  113. }
  114. SC_ScriptError("GetSoundOffset: Unknown sound name\n");
  115. return 0;
  116. }
  117. //==========================================================================
  118. //
  119. // SN_InitSequenceScript
  120. //
  121. //==========================================================================
  122. void SN_InitSequenceScript(void)
  123. {
  124. int i, j;
  125. int inSequence;
  126. int *tempDataStart;
  127. int *tempDataPtr;
  128. inSequence = -1;
  129. ActiveSequences = 0;
  130. for(i = 0; i < SS_MAX_SCRIPTS; i++)
  131. {
  132. SequenceData[i] = NULL;
  133. }
  134. SC_Open(SS_SCRIPT_NAME);
  135. while(SC_GetString())
  136. {
  137. if(*sc_String == ':')
  138. {
  139. if(inSequence != -1)
  140. {
  141. SC_ScriptError("SN_InitSequenceScript: Nested Script Error");
  142. }
  143. tempDataStart = (int *)Z_Malloc(SS_TEMPBUFFER_SIZE,
  144. PU_STATIC, NULL);
  145. memset(tempDataStart, 0, SS_TEMPBUFFER_SIZE);
  146. tempDataPtr = tempDataStart;
  147. for(i = 0; i < SS_MAX_SCRIPTS; i++)
  148. {
  149. if(SequenceData[i] == NULL)
  150. {
  151. break;
  152. }
  153. }
  154. if(i == SS_MAX_SCRIPTS)
  155. {
  156. I_Error("Number of SS Scripts >= SS_MAX_SCRIPTS");
  157. }
  158. for(j = 0; j < SEQ_NUMSEQ; j++)
  159. {
  160. if(!strcasecmp(SequenceTranslate[j].name, sc_String+1))
  161. {
  162. SequenceTranslate[j].scriptNum = i;
  163. inSequence = j;
  164. break;
  165. }
  166. }
  167. continue; // parse the next command
  168. }
  169. if(inSequence == -1)
  170. {
  171. continue;
  172. }
  173. if(SC_Compare(SS_STRING_PLAYUNTILDONE))
  174. {
  175. VerifySequencePtr(tempDataStart, tempDataPtr);
  176. SC_MustGetString();
  177. *tempDataPtr++ = SS_CMD_PLAY;
  178. *tempDataPtr++ = GetSoundOffset(sc_String);
  179. *tempDataPtr++ = SS_CMD_WAITUNTILDONE;
  180. }
  181. else if(SC_Compare(SS_STRING_PLAY))
  182. {
  183. VerifySequencePtr(tempDataStart, tempDataPtr);
  184. SC_MustGetString();
  185. *tempDataPtr++ = SS_CMD_PLAY;
  186. *tempDataPtr++ = GetSoundOffset(sc_String);
  187. }
  188. else if(SC_Compare(SS_STRING_PLAYTIME))
  189. {
  190. VerifySequencePtr(tempDataStart, tempDataPtr);
  191. SC_MustGetString();
  192. *tempDataPtr++ = SS_CMD_PLAY;
  193. *tempDataPtr++ = GetSoundOffset(sc_String);
  194. SC_MustGetNumber();
  195. *tempDataPtr++ = SS_CMD_DELAY;
  196. *tempDataPtr++ = sc_Number;
  197. }
  198. else if(SC_Compare(SS_STRING_PLAYREPEAT))
  199. {
  200. VerifySequencePtr(tempDataStart, tempDataPtr);
  201. SC_MustGetString();
  202. *tempDataPtr++ = SS_CMD_PLAYREPEAT;
  203. *tempDataPtr++ = GetSoundOffset(sc_String);
  204. }
  205. else if(SC_Compare(SS_STRING_DELAY))
  206. {
  207. VerifySequencePtr(tempDataStart, tempDataPtr);
  208. *tempDataPtr++ = SS_CMD_DELAY;
  209. SC_MustGetNumber();
  210. *tempDataPtr++ = sc_Number;
  211. }
  212. else if(SC_Compare(SS_STRING_DELAYRAND))
  213. {
  214. VerifySequencePtr(tempDataStart, tempDataPtr);
  215. *tempDataPtr++ = SS_CMD_DELAYRAND;
  216. SC_MustGetNumber();
  217. *tempDataPtr++ = sc_Number;
  218. SC_MustGetNumber();
  219. *tempDataPtr++ = sc_Number;
  220. }
  221. else if(SC_Compare(SS_STRING_VOLUME))
  222. {
  223. VerifySequencePtr(tempDataStart, tempDataPtr);
  224. *tempDataPtr++ = SS_CMD_VOLUME;
  225. SC_MustGetNumber();
  226. *tempDataPtr++ = sc_Number;
  227. }
  228. else if(SC_Compare(SS_STRING_END))
  229. {
  230. int dataSize;
  231. *tempDataPtr++ = SS_CMD_END;
  232. dataSize = (tempDataPtr-tempDataStart)*sizeof(int);
  233. SequenceData[i] = (int *)Z_Malloc(dataSize, PU_STATIC,
  234. NULL);
  235. memcpy(SequenceData[i], tempDataStart, dataSize);
  236. Z_Free(tempDataStart);
  237. inSequence = -1;
  238. }
  239. else if(SC_Compare(SS_STRING_STOPSOUND))
  240. {
  241. SC_MustGetString();
  242. SequenceTranslate[inSequence].stopSound =
  243. GetSoundOffset(sc_String);
  244. *tempDataPtr++ = SS_CMD_STOPSOUND;
  245. }
  246. else
  247. {
  248. SC_ScriptError("SN_InitSequenceScript: Unknown commmand.\n");
  249. }
  250. }
  251. }
  252. //==========================================================================
  253. //
  254. // SN_StartSequence
  255. //
  256. //==========================================================================
  257. void SN_StartSequence(mobj_t *mobj, int sequence)
  258. {
  259. seqnode_t *node;
  260. SN_StopSequence(mobj); // Stop any previous sequence
  261. node = (seqnode_t *)Z_Malloc(sizeof(seqnode_t), PU_STATIC, NULL);
  262. node->sequencePtr = SequenceData[SequenceTranslate[sequence].scriptNum];
  263. node->sequence = sequence;
  264. node->mobj = mobj;
  265. node->delayTics = 0;
  266. node->stopSound = SequenceTranslate[sequence].stopSound;
  267. node->volume = 127; // Start at max volume
  268. if(!SequenceListHead)
  269. {
  270. SequenceListHead = node;
  271. node->next = node->prev = NULL;
  272. }
  273. else
  274. {
  275. SequenceListHead->prev = node;
  276. node->next = SequenceListHead;
  277. node->prev = NULL;
  278. SequenceListHead = node;
  279. }
  280. ActiveSequences++;
  281. return;
  282. }
  283. //==========================================================================
  284. //
  285. // SN_StartSequenceName
  286. //
  287. //==========================================================================
  288. void SN_StartSequenceName(mobj_t *mobj, char *name)
  289. {
  290. int i;
  291. for(i = 0; i < SEQ_NUMSEQ; i++)
  292. {
  293. if(!strcmp(name, SequenceTranslate[i].name))
  294. {
  295. SN_StartSequence(mobj, i);
  296. return;
  297. }
  298. }
  299. }
  300. //==========================================================================
  301. //
  302. // SN_StopSequence
  303. //
  304. //==========================================================================
  305. void SN_StopSequence(mobj_t *mobj)
  306. {
  307. seqnode_t *node;
  308. for(node = SequenceListHead; node; node = node->next)
  309. {
  310. if(node->mobj == mobj)
  311. {
  312. S_StopSound(mobj);
  313. if(node->stopSound)
  314. {
  315. S_StartSoundAtVolume(mobj, node->stopSound, node->volume);
  316. }
  317. if(SequenceListHead == node)
  318. {
  319. SequenceListHead = node->next;
  320. }
  321. if(node->prev)
  322. {
  323. node->prev->next = node->next;
  324. }
  325. if(node->next)
  326. {
  327. node->next->prev = node->prev;
  328. }
  329. Z_Free(node);
  330. ActiveSequences--;
  331. }
  332. }
  333. }
  334. //==========================================================================
  335. //
  336. // SN_UpdateActiveSequences
  337. //
  338. //==========================================================================
  339. void SN_UpdateActiveSequences(void)
  340. {
  341. seqnode_t *node;
  342. boolean sndPlaying;
  343. if(!ActiveSequences || paused)
  344. { // No sequences currently playing/game is paused
  345. return;
  346. }
  347. for(node = SequenceListHead; node; node = node->next)
  348. {
  349. if(node->delayTics)
  350. {
  351. node->delayTics--;
  352. continue;
  353. }
  354. sndPlaying = S_GetSoundPlayingInfo(node->mobj, node->currentSoundID);
  355. switch(*node->sequencePtr)
  356. {
  357. case SS_CMD_PLAY:
  358. if(!sndPlaying)
  359. {
  360. node->currentSoundID = *(node->sequencePtr+1);
  361. S_StartSoundAtVolume(node->mobj, node->currentSoundID,
  362. node->volume);
  363. }
  364. node->sequencePtr += 2;
  365. break;
  366. case SS_CMD_WAITUNTILDONE:
  367. if(!sndPlaying)
  368. {
  369. node->sequencePtr++;
  370. node->currentSoundID = 0;
  371. }
  372. break;
  373. case SS_CMD_PLAYREPEAT:
  374. if(!sndPlaying)
  375. {
  376. node->currentSoundID = *(node->sequencePtr+1);
  377. S_StartSoundAtVolume(node->mobj, node->currentSoundID,
  378. node->volume);
  379. }
  380. break;
  381. case SS_CMD_DELAY:
  382. node->delayTics = *(node->sequencePtr+1);
  383. node->sequencePtr += 2;
  384. node->currentSoundID = 0;
  385. break;
  386. case SS_CMD_DELAYRAND:
  387. node->delayTics = *(node->sequencePtr+1)+
  388. M_Random()%(*(node->sequencePtr+2)-*(node->sequencePtr+1));
  389. node->sequencePtr += 2;
  390. node->currentSoundID = 0;
  391. break;
  392. case SS_CMD_VOLUME:
  393. node->volume = (127*(*(node->sequencePtr+1)))/100;
  394. node->sequencePtr += 2;
  395. break;
  396. case SS_CMD_STOPSOUND:
  397. // Wait until something else stops the sequence
  398. break;
  399. case SS_CMD_END:
  400. SN_StopSequence(node->mobj);
  401. break;
  402. default:
  403. break;
  404. }
  405. }
  406. }
  407. //==========================================================================
  408. //
  409. // SN_StopAllSequences
  410. //
  411. //==========================================================================
  412. void SN_StopAllSequences(void)
  413. {
  414. seqnode_t *node;
  415. for(node = SequenceListHead; node; node = node->next)
  416. {
  417. node->stopSound = 0; // don't play any stop sounds
  418. SN_StopSequence(node->mobj);
  419. }
  420. }
  421. //==========================================================================
  422. //
  423. // SN_GetSequenceOffset
  424. //
  425. //==========================================================================
  426. int SN_GetSequenceOffset(int sequence, int *sequencePtr)
  427. {
  428. return (sequencePtr-SequenceData[SequenceTranslate[sequence].scriptNum]);
  429. }
  430. //==========================================================================
  431. //
  432. // SN_ChangeNodeData
  433. //
  434. // nodeNum zero is the first node
  435. //==========================================================================
  436. void SN_ChangeNodeData(int nodeNum, int seqOffset, int delayTics, int volume,
  437. int currentSoundID)
  438. {
  439. int i;
  440. seqnode_t *node;
  441. i = 0;
  442. node = SequenceListHead;
  443. while(node && i < nodeNum)
  444. {
  445. node = node->next;
  446. i++;
  447. }
  448. if(!node)
  449. { // reach the end of the list before finding the nodeNum-th node
  450. return;
  451. }
  452. node->delayTics = delayTics;
  453. node->volume = volume;
  454. node->sequencePtr += seqOffset;
  455. node->currentSoundID = currentSoundID;
  456. }