R-PAGE
Resistance's Portable-Adventure-Game-Engine
sound.c
Go to the documentation of this file.
1 #ifdef LATTICE
2 /*
3  EASY-SOUND V2.00 1990-09-23 ANDERS BJERIN
4  ADPCM Decoder by Kalms
5 */
6 
7 /* Include some important header files: */
8 #include <assert.h>
9 #include "rpage/aos/inc.prl"
10 #include <exec/types.h>
11 #include <exec/memory.h>
12 #include <devices/audio.h>
13 #include <stdio.h>
14 #include <string.h>
15 
16 #include "ext/tinfl.h"
17 #include "ext/aos/shrinkler.h"
18 #include "ext/aos/nrv2.h"
19 
20 #include "rpage/aos/sound.h"
21 #include "rpage/err.h"
22 #include "rpage/aos/adpcm.h"
23 
24 #define CLOCK_CONSTANT 3579545
25 #define MUSIC_PRIORITY 0
26 
27 extern struct DosLibrary *DOSBase;
28 
29 /* An IOAudio pointer to each sound channel: */
30 struct IOAudio *IOA[4] = {NULL, NULL, NULL, NULL};
31 
45 SoundInfo *LoadPackedSound(char *filename, BYTE *packed_block, BYTE *unpacked_block)
46 {
47 #ifdef DEBUG_ENABLE_AUDIO
48  ULONG encoded_block_size = 0, unpacked_block_size = 0, packed_block_size = 0;
49  ULONG frequency = 8000;
50  BPTR fileHandle;
51  char encoder_tag[4], compressor_tag[4], tag[4];
52  UWORD mod_size;
53  BYTE *encoded_block = NULL;
54  SoundInfo *sound;
55 #ifdef DEBUG_MACROS
56  printf("LoadPackedSound(%s)\n", filename);
57 #endif
58  if ((fileHandle = Open(filename, MODE_OLDFILE)))
59  {
60  Read(fileHandle, &tag, 4);
61  if (strncmp(tag, "SMPK", 4) == 0)
62  {
63  // read the sample length
64  Read(fileHandle, &tag, 4); // SIZE
65  if (strncmp(tag, "SIZE", 4) == 0)
66  {
67  Read(fileHandle, &unpacked_block_size, 4);
68 #ifdef DEBUG_MACROS
69  printf("unpacked_block_size = %d\n", unpacked_block_size);
70 #endif
71  // read the replay frequency
72  Read(fileHandle, &tag, 4); // FREQ
73  Read(fileHandle, &frequency, 4);
74 
75  // read the encoder name
76  Read(fileHandle, &encoder_tag, 4);
77 #ifdef DEBUG_MACROS
78  printf("encoder found : %c%c%c%c.\n", encoder_tag[0],encoder_tag[1],encoder_tag[2],encoder_tag[3]);
79 #endif
80  Read(fileHandle, &tag, 4);
81  if (strncmp(tag, "SIZE", 4) == 0)
82  {
83  Read(fileHandle, &encoded_block_size, 4);
84 #ifdef DEBUG_MACROS
85  printf("encoded_block_size = %d\n", encoded_block_size);
86 #endif
87  // read the compressor name
88  Read(fileHandle, &compressor_tag, 4);
89 #ifdef DEBUG_MACROS
90  printf("compressor found : %c%c%c%c.\n", compressor_tag[0],compressor_tag[1],compressor_tag[2],compressor_tag[3]);
91 #endif
92  Read(fileHandle, &tag, 4);
93  if (strncmp(tag, "SIZE", 4) == 0)
94  {
95  Read(fileHandle, &packed_block_size, 4);
96 #ifdef DEBUG_MACROS
97  printf("packed_block_size = %d\n", packed_block_size);
98 #endif
99  Read(fileHandle, packed_block, packed_block_size);
100  }
101  else
102  printf(err_no_size_found); // compressor
103  }
104  else
105  printf(err_no_size_found); // encoder
106  }
107  else
108  printf(err_no_size_found); // original sample
109 
110  if ((encoded_block_size > 0) && (unpacked_block_size > 0) && (packed_block_size > 0))
111  {
112  // printf("unpacked_block = %x\n", unpacked_block);
113  if (unpacked_block == NULL)
114  unpacked_block = AllocMem(unpacked_block_size, MEMF_CHIP);
115 
116  encoded_block = (UBYTE *)calloc(encoded_block_size, sizeof(UBYTE));
117  assert(unpacked_block != NULL);
118 
119  if (encoded_block != NULL)
120  {
121  // printf("unpacked_block (post malloc) = %x\n", unpacked_block);
122 
123  // printf("compressor_tag = %c%c%c%c\n", compressor_tag[0], compressor_tag[1], compressor_tag[2], compressor_tag[3]);
124  if (strncmp(compressor_tag, "MINZ", 4) == 0)
125  tinfl_decompress_mem_to_mem(encoded_block, encoded_block_size, packed_block, packed_block_size, 1);
126  else if (strncmp(compressor_tag, "SHRK", 4) == 0)
127  ShrinklerDecompress(packed_block, encoded_block, NULL, NULL);
128  else if (strncmp(compressor_tag, "NRV2", 4) == 0)
129  nrv2s_unpack(packed_block, encoded_block);
130 
131  if (strncmp(encoder_tag, "ADPC", 4) == 0)
132  {
133  adpcm_decode(encoded_block, encoded_block_size, unpacked_block);
134  }
135  else if (strncmp(encoder_tag, "GLI2", 4) == 0)
136  {
137  printf("Gligli MDPCM not supported yet!\n");
138  }
139  else if (strncmp(encoder_tag, "8SVX", 4) == 0)
140  {
141  memcpy(unpacked_block, encoded_block, unpacked_block_size);
142  }
143 
144  sound = (SoundInfo *)calloc(1, sizeof(SoundInfo));
145  assert(sound != NULL);
146  sound->SoundBuffer = unpacked_block;
147  sound->FileLength = unpacked_block_size;
148  sound->RecordRate = frequency;
149 
150  free(encoded_block);
151  encoded_block = NULL;
152 
153  return sound;
154  }
155  else
156  {
157  printf("LoadPackedSound(), encoded_block calloc() failed!\n");
158  return NULL;
159  }
160 
161 
162  }
163  }
164  else
165  {
166  printf("!Not a SMPK File!\n");
167  }
168  }
169 
170  return NULL;
171 #else
172  return NULL;
173 #endif
174 }
175 
176 /* Declare the functions we are going to use: */
177 // SoundInfo *PrepareSound(STRPTR file);
178 // BOOL PlaySound(SoundInfo *info, UWORD volume, UBYTE channel,
179 // WORD delta_rate, UWORD repeat);
180 // void StopSound(UBYTE channel);
181 // void RemoveSound();
182 
183 // BOOL PrepareIOA(UWORD period, UWORD volume, UWORD cycles, UBYTE channel,
184 // SoundInfo *info);
185 // UWORD LoadSound(STRPTR filename, SoundInfo *info);
186 // ULONG GetSize(STRPTR filename);
187 // ULONG SizeIFF(STRPTR filename);
188 // UWORD ReadIFF(STRPTR filename, SoundInfo *info);
189 // BOOL MoveTo(STRPTR check_string, FILE *file_ptr);
190 
204 
205 SoundInfo *PrepareSound(STRPTR file)
206 {
207 #ifdef DEBUG_ENABLE_AUDIO
208  /* Declare a pointer to a SoundInfo structure: */
209  SoundInfo *info = NULL;
210 
211  /* Allocate memory for a SoundInfo structure: (The memory can be of */
212  /* any type, and should be cleared. */
213  info = (SoundInfo *)AllocMem(sizeof(SoundInfo), MEMF_PUBLIC | MEMF_CLEAR);
214 
215  if (info != NULL)
216  {
217  /* The memory have been successfully allocated. */
218 
219  /* Get the size of the file, and store it in the SoundInfo struct.: */
220  if (info->FileLength = GetSize(file))
221  {
222  /* Allocate enough memory for the sampled sound, and store a */
223  /* pointer to the buffer in the SoundInfo structure: */
224  info->SoundBuffer =
225  (BYTE *)AllocMem(info->FileLength, MEMF_CHIP | MEMF_CLEAR);
226 
227  if (info->SoundBuffer)
228  {
229  /* The memory have been successfully allocated. */
230 
231  /* Load the sound, and store the record rate in the SoundInfo */
232  /* structure. If the sound could not be loaded, 0 is returned: */
233  if (info->RecordRate = LoadSound(file, info))
234  {
235  /* OK! The sound has successfully been loaded. */
236 
237  /* Old FutureSound files were saved in kHz. If the record rate */
238  /* is less than one hundered, we know it is an old FutureSound */
239  /* file, and simply multiply the rate with one thousand: */
240  /* Astrofra : removed suspicious patch inherited from the past century
241  */
242  /* if( info->RecordRate < 100 )
243  info->RecordRate *= 1000;
244  */
245 
246  /* Return a pointer to the SoundInfo structure. (We return a */
247  /* normal memory pointer.) */
248  return (info); // return( (CPTR) info ); /* OK! */
249  }
250  else
251  {
252  /* ERROR! We could not load the sound! */
253 
254  /* Deallocate the memory for the sound buffer: */
255  FreeMem(info->SoundBuffer, info->FileLength);
256  }
257  }
258  }
259  /* Deallocate the memory the SoundInfo structure: */
260  FreeMem(info, sizeof(SoundInfo));
261  }
262 
263  /* We have not been able to prepare the sound. All allocated memory */
264  /* have been deallocated, and we return NULL. */
265  return NULL; /* ERROR! */
266 #else
267  return NULL;
268 #endif
269 }
270 
271 /* PlaySound() */
272 /* PlaySound() plays one already prepared sound effect. You can decide */
273 /* what volume, which channel should, what rate, and how many times the */
274 /* sound should be played. */
275 /* */
276 /* Synopsis: ok = PlaySound( pointer, volume, channel, drate, times ); */
277 /* ok: (BOOL) If the sound was played successfully TRUE is */
278 /* returned, else FALSE. */
279 /* pointer: (CPTR) Actually a pointer to a SoundInfo structure. This */
280 /* pointer was returned by PrepareSound(). */
281 /* volume: (UWORD) Volume, 0 to 64. */
282 /* channel: (UBYTE) Which channel should be used. (LEFT0, RIGHT0, */
283 /* RIGHT1 or LEFT1) */
284 /* drate: (WORD) Delta rate. When the sound is prepared, the record */
285 /* rate is automatically stored in the SoundInfo structure, */
286 /* so if you do not want to change the rate, write 0. */
287 /* times: (UWORD) How many times the sound should be played. If you */
288 /* want to play the sound forever, write 0. (To stop a sound */
289 /* call the function StopSound().) */
290 
291 BOOL PlaySound(SoundInfo *info, UWORD volume, UBYTE channel,
292  WORD delta_rate, UWORD repeat)
293 {
294 #ifdef DEBUG_ENABLE_AUDIO
295  /* Before we may play the sound, we must make sure that the sound is */
296  /* not already being played. We will therefore call the function */
297  /* StopSound(), in order to stop the sound if it is playing: */
298  StopSound(channel);
299 
300  /* Call the PrepareIOA() function that will declare and initialize an */
301  /* IOAudio structure: */
302  if (PrepareIOA(CLOCK_CONSTANT / info->RecordRate + delta_rate, volume, repeat,
303  channel, info))
304  {
305  /* We will now start playing the sound: */
306  BeginIO((struct IORequest *)IOA[channel]);
307 #ifdef DEBUG_MACROS
308  printf("PlaySound(), IOA[%d] = %x\n", channel, IOA[channel]);
309 #endif
310  return TRUE; /* OK! */
311  }
312  else
313  return FALSE; /* ERROR! */
314 #else
315  return FALSE;
316 #endif
317 }
318 
319 /* StopSound() */
320 /* StopSound() will stop the specified audio channel from continuing */
321 /* to play the sound. It will also close all devices and ports that */
322 /* have been opened, and deallocate some memory that have been */
323 /* allocated. */
324 /* */
325 /* Synopsis: StopSound( channel ); */
326 /* channel: (UBYTE) The audio channel that should be stopped. (LEFT0, */
327 /* LEFT1, RIGHT0 or RIGHT1.) */
328 
329 void StopSound(UBYTE channel)
330 {
331 #ifdef DEBUG_ENABLE_AUDIO
332  /* Check if the IOAudio structure exist: */
333  if (IOA[channel])
334  {
335  /* 1. Stop the sound: */
336  AbortIO((struct IORequest *)IOA[channel]);
337 
338  /* 2. If there exist a Sound Device, close it: */
339  if (IOA[channel]->ioa_Request.io_Device)
340  CloseDevice((struct IORequest *)IOA[channel]);
341 
342  /* 3. If there exist a Message Port, delete it: */
343  if (IOA[channel]->ioa_Request.io_Message.mn_ReplyPort)
344  DeletePort(IOA[channel]->ioa_Request.io_Message.mn_ReplyPort);
345 #ifdef DEBUG_MACROS
346  printf("StopSound(), IOA[%d] = %x\n", channel, IOA[channel]);
347 #endif
348  if (IOA[channel] != NULL)
349  {
350  FreeMem(IOA[channel], sizeof(struct IOAudio));
351  IOA[channel] = NULL;
352  }
353  }
354 #endif
355 }
356 
357 /* RemoveSound() */
358 /* RemoveSound() will stop playing the sound, and deallocate all memory */
359 /* that was allocated by the PrepareSound() function. Before your */
360 /* program terminates, all sound that has been prepared, MUST be */
361 /* removed. */
362 /* */
363 /* IMPORTANT! The each channel that is currently playing the sound must */
364 /* be stopped! (Use the StopSound() function.) */
365 
366 /* Synopsis: RemoveSound( pointer ); */
367 /* pointer: (CPTR) Actually a pointer to a SoundInfo structure. */
368 
369 void RemoveSound(SoundInfo *info)
370 {
371 #ifdef DEBUG_ENABLE_AUDIO
372  /* IMPORTANT! The sound must have been */
373  /* stopped before you may remove it!!! */
374 
375  /* Have we allocated a SoundInfo structure? */
376  if (info != NULL)
377  {
378  /* Deallocate the sound buffer: */
379 #ifdef DEBUG_MACROS
380  printf("RemoveSound(), SoundBuffer = %x, FileLength = %d\n", info->SoundBuffer, info->FileLength);
381 #endif
382  if (info->SoundBuffer != NULL)
383  {
384  FreeMem(info->SoundBuffer, info->FileLength);
385  info->SoundBuffer = NULL;
386  }
387 
388  /* Deallocate the SoundInfo structure: */
389  free(info);
390  info = NULL;
391  // FreeMem(info, sizeof(SoundInfo));
392  // info = NULL;
393  }
394 #endif
395 }
396 
397 void RemoveSoundStruct(SoundInfo *info)
398 {
399 #ifdef DEBUG_ENABLE_AUDIO
400  /* IMPORTANT! The sound must have been */
401  /* stopped before you may remove it!!! */
402  /* Have we allocated a SoundInfo structure? */
403  if (info != NULL)
404  {
405  /* Deallocate the SoundInfo structure: */
406  // FreeMem(info, sizeof(SoundInfo));
407  free(info);
408  info = NULL;
409  }
410 #endif
411 }
412 
413 /* PrepareIOA() */
414 /* PrepareIOA() allocates and initializes an IOAudio structure. */
415 /* */
416 /* Synopsis: ok = PrepareIOA( period, volume, cycles, channel, pointer ); */
417 /* */
418 /* ok: (BOOL) If the IOAudio structure was allocated and */
419 /* initialized successfully, TRUE is returned, else FALSE. */
420 /* period: (UWORD) Period time. */
421 /* volume: (UWORD) Volume, 0 to 64. */
422 /* cycles: (UWORD) How many times the sound should be played. */
423 /* (0 : forever) */
424 /* channel: (UBYTE) Which channel should be used. (LEFT0, RIGHT0, */
425 /* RIGHT1 or LEFT1) */
426 /* pointer: (CPTR) Actually a pointer to a SoundInfo structure. */
427 
428 BOOL PrepareIOA(UWORD period, UWORD volume, UWORD cycles, UBYTE channel, SoundInfo *info)
429 {
430 #ifdef DEBUG_ENABLE_AUDIO
431  UBYTE ch;
432 
433  /* Declare a pointer to a MsgPort structure: */
434  struct MsgPort *port;
435 
436  /* Allocate space for an IOAudio structure: */
437  IOA[channel] = (struct IOAudio *)AllocMem(sizeof(struct IOAudio),
438  MEMF_PUBLIC | MEMF_CLEAR);
439 
440  /* Could we allocate enough memory? */
441  if (IOA[channel])
442  {
443  /* Create Message port: */
444  if ((port = (struct MsgPort *)CreatePort("Sound Port", 0)) == NULL)
445  {
446  /* ERROR! Could not create message port! */
447  /* Deallocate the IOAudio structure: */
448  if (IOA[channel] != NULL)
449  {
450  FreeMem(IOA[channel], sizeof(struct IOAudio));
451  IOA[channel] = NULL;
452  }
453 
454  return (FALSE); /* ERROR! */
455  }
456  else
457  {
458  /* Port created successfully! */
459  /* Initialize the IOAudion structure: */
460 
461  /* Priority: */
462  IOA[channel]->ioa_Request.io_Message.mn_Node.ln_Pri = MUSIC_PRIORITY;
463 
464  /* Port: */
465  IOA[channel]->ioa_Request.io_Message.mn_ReplyPort = port;
466 
467  /* Channel: */
468  ch = 1 << channel;
469  IOA[channel]->ioa_Data = &ch;
470 
471  /* Length: */
472  IOA[channel]->ioa_Length = sizeof(UBYTE);
473 
474  /* Open Audio Device: */
475  if (OpenDevice(AUDIONAME, 0, (struct IORequest *)IOA[channel], 0))
476  {
477  /* ERROR! Could not open the device! */
478  /* Delete Sound Port: */
479  DeletePort(port);
480 
481  /* Deallocate the IOAudio structure: */
482  if (IOA[channel] != NULL)
483  {
484  FreeMem(IOA[channel], sizeof(struct IOAudio));
485  IOA[channel] = NULL;
486  }
487 
488  return (FALSE); /* ERROR! */
489  }
490  else
491  {
492  /* Device opened successfully! */
493  /* Initialize the rest of the IOAudio structure: */
494  IOA[channel]->ioa_Request.io_Flags = ADIOF_PERVOL;
495  IOA[channel]->ioa_Request.io_Command = CMD_WRITE;
496  IOA[channel]->ioa_Period = period;
497  IOA[channel]->ioa_Volume = volume;
498  IOA[channel]->ioa_Cycles = cycles;
499 
500  /* The Audio Chip can of some strange reason not play sampled */
501  /* sound that is longer than 131KB. So if the sound is too long, */
502  /* we simply cut it off: */
503  if (info->FileLength > 131000)
504  IOA[channel]->ioa_Length = 131000;
505  else
506  IOA[channel]->ioa_Length = info->FileLength;
507 
508  // printf("PrepareIOA() ioa_Length = %d\n", IOA[ channel ]->ioa_Length);
509 
510  IOA[channel]->ioa_Data = info->SoundBuffer;
511 
512  return (TRUE); /* OK! */
513  }
514  }
515  }
516  return FALSE; /* ERROR! */
517 #else
518  return FALSE;
519 #endif
520 }
521 
522 /* LoadSound() */
523 /* LoadSound() will load sampled sound that was either saved in IFF or */
524 /* FutureSound format. */
525 /* */
526 /* Synopsis: rate = LoadSound( filename, pointer ); */
527 /* rate: (UWORD) The record rate is returned if the sound was */
528 /* successfully loaded, else 0. */
529 /* filename: (STRPTR) Pointer to a string containing the name of the */
530 /* sound file. For example�"df0:Explosion.snd". */
531 /* pointer: (CPTR) Actually a pointer to a SoundInfo structure. */
532 
533 UWORD LoadSound(STRPTR filename, SoundInfo *info)
534 {
535 #ifdef DEBUG_ENABLE_AUDIO
536  FILE *file_ptr; /* Pointer to a file. */
537  ULONG length; /* Data Length. */
538  UWORD record_rate; /* Record rate. */
539 
540  /* Check if it is an IFF File: */
541  if (SizeIFF(filename))
542  {
543  /* Yes, it is an IFF file. Read it: */
544  return (ReadIFF(filename, info));
545  }
546  else
547  {
548  /* No, then it is probably a FutureSound file. */
549  /* Open the file so we can read it: */
550  if ((file_ptr = fopen(filename, "r")) == 0)
551  return (0); /* ERROR! Could not open the file! */
552 
553  /* Read the data length: */
554  if (fread((char *)&length, sizeof(ULONG), 1, file_ptr) == 0)
555  {
556  /* ERROR! Could not read the data length! */
557  /* Close the file, and return zero: */
558  fclose(file_ptr);
559  return (0);
560  }
561 
562  /* Read the record rate: */
563  if (fread((char *)&record_rate, sizeof(UWORD), 1, file_ptr) == 0)
564  {
565  /* ERROR! Could not read the record rate! */
566  /* Close the file, and return zero: */
567  fclose(file_ptr);
568  return (0);
569  }
570 
571  /* Read the sampled sound data into the buffer: */
572  if (fread((char *)info->SoundBuffer, length, 1, file_ptr) == 0)
573  {
574  /* ERROR! Could not read the data! */
575  /* Close the file, and return zero: */
576  fclose(file_ptr);
577  return (0);
578  }
579 
580  /* Close the file: */
581  fclose(file_ptr);
582 
583  /* Return the record rate: */
584  // printf("LoadSound() record_rate = %d\n", record_rate);
585  return (record_rate);
586  }
587 #else
588  return NULL;
589 #endif
590 }
591 
592 /* GetSize() */
593 /* GetSize() returns the size of the file which was saved in either */
594 /* IFF or FutureSound format. */
595 /* */
596 /* Synopsis: length = GetSize( filename ); */
597 /* length: (ULONG) Data length. */
598 /* filename: (STRPTR) Pointer to a string containing the name of the */
599 /* sound file. For example�"df0:Explosion.snd". */
600 
601 ULONG GetSize(STRPTR filename)
602 {
603 #ifdef DEBUG_ENABLE_AUDIO
604  FILE *file_ptr; /* Pointer to a file. */
605  ULONG length; /* Data length. */
606 
607  /* Check if it is an IFF File: */
608  if ((length = SizeIFF(filename)) == 0)
609  {
610  /* No, then it is probably a FutureSound file. */
611  /* Open the file so we can read it: */
612  if ((file_ptr = fopen(filename, "r")) == 0)
613  return (0); /* ERROR! Could not open the file! */
614 
615  /* Read the data length: */
616  if (fread((char *)&length, sizeof(ULONG), 1, file_ptr) == 0)
617  {
618  /* ERROR! Could not read the data length! */
619  /* Close the file, and return zero: */
620  fclose(file_ptr);
621  return (0);
622  }
623 
624  /* Close the file: */
625  fclose(file_ptr);
626  }
627  // printf("GetSize() length = %d\n", length);
628  return (length);
629 #else
630  return 0;
631 #endif
632 }
633 
634 /* SizeIFF() */
635 /* SizeIFF() returns the size of an IFF file, or zero if something */
636 /* went wrong (for example, It was not an IFF file). */
637 /* */
638 /* Synopsis: length = SizeIFF( filename ); */
639 /* length: (ULONG) Data length. */
640 /* filename: (STRPTR) Pointer to a string containing the name of the */
641 /* IFF file. For example�"df0:Explosion.snd". */
642 
643 ULONG SizeIFF(STRPTR filename)
644 {
645 #ifdef DEBUG_ENABLE_AUDIO
646  FILE *file_ptr; /* Pointer to a file. */
647  STRPTR empty_string = " "; /* Four spaces. */
648  LONG dummy; /* A dummy variable. */
649  Voice8Header Header; /* Voice8Header structure. */
650 
651  /* Try to open the file: */
652  if (file_ptr = fopen(filename, "r"))
653  {
654  fread((char *)empty_string, 4, 1, file_ptr);
655  if (strcmp(empty_string, "FORM") == 0)
656  {
657  /* Read twice: */
658  fread((char *)empty_string, 4, 1, file_ptr);
659  fread((char *)empty_string, 4, 1, file_ptr);
660 
661  /* Check if it is a "8SVX" file, or not: */
662  if (strcmp(empty_string, "8SVX") == 0)
663  {
664  MoveTo("VHDR", file_ptr);
665  fread((char *)&dummy, sizeof(LONG), 1, file_ptr);
666  fread((char *)&Header, sizeof(Header), 1, file_ptr);
667 
668  /* Close the file, and return the length: */
669  fclose(file_ptr);
670 
671  // printf("SizeIFF() oneShotHiSamples + repeatHiSamples = %d\n",
672  // Header.oneShotHiSamples + Header.repeatHiSamples);
673  return (Header.oneShotHiSamples + Header.repeatHiSamples);
674  }
675  }
676  /* Close the file: */
677  fclose(file_ptr);
678  }
679  /* Return zero: (ERROR) */
680  return (0);
681 #else
682  return 0;
683 #endif
684 }
685 
686 /* ReadIFF() */
687 /* ReadIFF() reads an IFF file into the buffer, and returns the record */
688 /* rate. */
689 /* */
690 /* Synopsis: rate = ReadIFF( filename, pointer ); */
691 /* rate: (UWORD) The record rate is returned if the sound was */
692 /* successfully loaded, else 0. */
693 /* filename: (STRPTR) Pointer to a string containing the name of the */
694 /* sound file. For example�"df0:Explosion.snd". */
695 /* pointer: (CPTR) Actually a pointer to a SoundInfo structure. */
696 
697 UWORD ReadIFF(STRPTR filename, SoundInfo *info)
698 {
699 #ifdef DEBUG_ENABLE_AUDIO
700  FILE *file_ptr; /* Pointer to a file. */
701  STRPTR empty_string = " "; /* Four spaces. */
702  LONG dummy; /* A dummy variable. */
703  Voice8Header Header; /* Voice8Header structure. */
704 
705  /* Try to open the file: */
706  if (file_ptr = fopen(filename, "r"))
707  {
708  fread((char *)empty_string, 4, 1, file_ptr);
709  if (strcmp(empty_string, "FORM") == 0)
710  {
711  /* Read twice: */
712  fread((char *)empty_string, 4, 1, file_ptr);
713  fread((char *)empty_string, 4, 1, file_ptr);
714 
715  /* Check if it is a "8SVX" file, or not: */
716  if (strcmp(empty_string, "8SVX") == 0)
717  {
718  MoveTo("VHDR", file_ptr);
719  fread((char *)&dummy, sizeof(LONG), 1, file_ptr);
720  fread((char *)&Header, sizeof(Header), 1, file_ptr);
721 
722  MoveTo("BODY", file_ptr);
723  fread((char *)&dummy, sizeof(LONG), 1, file_ptr);
724  fread((char *)info->SoundBuffer,
725  Header.oneShotHiSamples + Header.repeatHiSamples, 1, file_ptr);
726 
727  /* Close the file, and return the record rate: */
728  fclose(file_ptr);
729 
730  // printf("ReadIFF() samplesPerSec = %d\n", Header.samplesPerSec);
731  return (Header.samplesPerSec);
732  }
733  }
734  /* Close the file: */
735  fclose(file_ptr);
736  }
737  /* Return zero: (ERROR) */
738  return (0);
739 #else
740  return 0;
741 #endif
742 }
743 
744 /* MoveTo() */
745 /* MoveTo() walks through an IFF file, and looks for chunks. */
746 /* */
747 /* Synopsis: MoveTo( chunk, file_ptr ); */
748 /* chunk: (STRPTR) The chunk we want to get to. */
749 /* file_ptr: (FILE *) Pointer to an already opened file. */
750 
751 BOOL MoveTo(STRPTR check_string, FILE *file_ptr)
752 {
753 #ifdef DEBUG_ENABLE_AUDIO
754  STRPTR empty_string = " "; /* Four spaces. */
755  int skip, loop; /* How much data should be skiped. */
756  LONG dummy; /* A dummy variable. */
757 
758  /* As long as we have not reached the EOF, continue: */
759  while (!feof(file_ptr))
760  {
761  fread((char *)empty_string, 4, 1, file_ptr);
762 
763  /* Have we found the right chunk? */
764  if (strcmp(check_string, empty_string) == 0)
765  return (0); /* YES! Return nothing. */
766 
767  /* Move foreward: */
768  fread((char *)&skip, sizeof(LONG), 1, file_ptr);
769  for (loop = 0; loop < skip; loop++)
770  fread((char *)&dummy, 1, 1, file_ptr);
771  }
772 #else
773  return FALSE;
774 #endif
775 }
776 
777 // void adpcm_decode(CodecState *state, UBYTE *input, int numSamples, UBYTE *output)
778 BYTE *adpcm_decode(UBYTE *Source, int Length, BYTE *Destination)
779 {
780 #ifdef DEBUG_ENABLE_AUDIO
781  const ULONG JoinCode = 0;
782  WORD EstMax = (WORD)(JoinCode & 0xffff);
783  UWORD Delta = (UWORD)((JoinCode & 0xffff0000) >> 16);
784  ULONG lDelta = 0;
785  const UBYTE Bits = 4;
786 
787  if(!Delta) Delta = 5;
788 
789  Length /= 3;
790 
791  while(Length--) {
792  UBYTE sampleCount = 24/Bits;
793  ULONG temp = (Source[0] << 16) | (Source[1] << 8) | Source[2];
794  Source+=3;
795 
796  while(sampleCount--) {
797  WORD newEstMax = (Delta >> 1);
798  UBYTE Shifter = (temp >> sampleCount*Bits);
799  UBYTE b = (Shifter & bitmask[Bits-1]);
800 
801  if ((Bits == 4) && ((Shifter & 0xf) == 0))
802  Delta = 4;
803 
804  while(b--) {
805  newEstMax += Delta;
806  }
807 
808  lDelta = Delta * Matrix[Bits-2][Shifter & bitmask[Bits-1]];
809 
810  if(Shifter & (1<<(Bits-1))) { // SignBit
811  newEstMax = -newEstMax;
812  }
813  EstMax = (EstMax + newEstMax) & 0xffff;
814 
815  Delta = (UWORD)((LONG)(lDelta + 8192) >> 14);
816 
817  if(Delta < 5) Delta = 5;
818 
819  newEstMax = EstMax >> 6;
820  if(127 < newEstMax)
821  *Destination++ = 127;
822  else if( -128 > newEstMax) {
823  *Destination++ = -128;
824  }
825  else
826  *Destination++ = newEstMax;
827  }
828  }
829  // return (Delta<<16|(EstMax&0xffff));
830  return Destination;
831 #else
832  return NULL;
833 #endif
834 }
835 
836 #endif
#define FALSE
Definition: utils.h:41
const char * err_no_size_found
Definition: err.c:2
int BOOL
Definition: utils.h:32
unsigned char UBYTE
Definition: utils.h:20
unsigned short UWORD
Definition: utils.h:28
struct DosLibrary * DOSBase
size_t tinfl_decompress_mem_to_mem(void *pOut_buf, size_t out_buf_len, const void *pSrc_buf, size_t src_buf_len, int flags)
Definition: tinfl.c:544
unsigned char BYTE
Definition: utils.h:16
unsigned long ULONG
Definition: utils.h:24
#define TRUE
Definition: utils.h:37