austream.c
上传用户:sun1608
上传日期:2007-02-02
资源大小:6116k
文件大小:9k
源码类别:

流媒体/Mpeg4/MP4

开发平台:

Visual C++

  1. /**********************************************************************
  2. MPEG-4 Audio VM
  3. audio i/o streams (.au format)
  4. This software module was originally developed by
  5. Heiko Purnhagen (University of Hannover)
  6. and edited by
  7. in the course of development of the MPEG-2 NBC/MPEG-4 Audio standard
  8. ISO/IEC 13818-7, 14496-1,2 and 3. This software module is an
  9. implementation of a part of one or more MPEG-2 NBC/MPEG-4 Audio tools
  10. as specified by the MPEG-2 NBC/MPEG-4 Audio standard. ISO/IEC gives
  11. users of the MPEG-2 NBC/MPEG-4 Audio standards free license to this
  12. software module or modifications thereof for use in hardware or
  13. software products claiming conformance to the MPEG-2 NBC/ MPEG-4 Audio
  14. standards. Those intending to use this software module in hardware or
  15. software products are advised that this use may infringe existing
  16. patents. The original developer of this software module and his/her
  17. company, the subsequent editors and their companies, and ISO/IEC have
  18. no liability for use of this software module or modifications thereof
  19. in an implementation. Copyright is not released for non MPEG-2
  20. NBC/MPEG-4 Audio conforming products. The original developer retains
  21. full right to use the code for his/her own purpose, assign or donate
  22. the code to a third party and to inhibit third party from using the
  23. code for non MPEG-2 NBC/MPEG-4 Audio conforming products. This
  24. copyright notice must be included in all copies or derivative works.
  25. Copyright (c) 1999.
  26. Source file: austream.c
  27. $Id: austream.c,v 1.3 2002/07/08 15:03:29 mvillari Exp $
  28. Authors:
  29. HP    Heiko Purnhagen, Uni Hannover <purnhage@tnt.uni-hannover.de>
  30. NM    Nikolaus Meine, Uni Hannover <meine@tnt.uni-hannover.de>
  31. Changes:
  32. 16-sep-98   HP/NM   born, based on au_io.c
  33. 06-aug-99   HP      fixed bug in AuOpenRead() numSample
  34. **********************************************************************/
  35. /* Audio i/o streaming with stdin/stdout support. */
  36. /* Header format: .au (Sun audio file format, aka SND or AFsp) */
  37. /* Sample format: 16 bit twos complement, uniform quantisation */
  38. /* Data size set to -1 (=unknown) */
  39. /* Multi channel data is interleaved: l0 r0 l1 r1 ... */
  40. /* Total number of samples (over all channels) is used. */
  41. #include <stdio.h>
  42. #include <stdlib.h>
  43. #include <string.h>
  44. #include "austream.h"
  45. #include "common_m4a.h"
  46. #include "bitstream.h"
  47. /* ---------- declarations ---------- */
  48. #define min(a,b) ((a) < (b) ? (a) : (b))
  49. #define max(a,b) ((a) > (b) ? (a) : (b))
  50. /* ---------- declarations (structures) ---------- */
  51. struct AuStreamStruct /* audio stream handle */
  52. {
  53.   FILE *f; /* stream handle */
  54.   long currentSample; /* number of samples read/written */
  55.   int eof; /* eof/error flag */
  56.   int write; /* write (not read) flag */
  57. };
  58. /* ---------- variables ---------- */
  59. static int AUdebugLevel = 0; /* debug level */
  60. /* ---------- local functions ---------- */
  61. static void putshort (short x, AuStream *s)
  62. {
  63.   register int a;
  64.   putc((x>>8)&255,s->f);
  65.   a = putc(x&255,s->f);
  66.   if (a==EOF)
  67.     s->eof = 1;
  68. }
  69. static void putint (long x, AuStream *s)
  70. {
  71.   int a;
  72.   putc((x>>24)&255,s->f);
  73.   putc((x>>16)&255,s->f);
  74.   putc((x>>8)&255,s->f);
  75.   a = putc(x&255,s->f);
  76.   if (a==EOF)
  77.     s->eof = 1;
  78. }
  79. static short getshort (AuStream *s)
  80. {
  81.   register int a,b;
  82.   
  83.   if (s->eof)
  84.     return 0;
  85.   
  86.   a = getc(s->f);
  87.   b = getc(s->f);
  88.   if (b==EOF) {
  89.     s->eof = 1;
  90.     return 0;
  91.   }
  92.   return (a<<8)|b;
  93. }
  94. static long getint (AuStream *s)
  95. {
  96.   int a,b;
  97.   if (s->eof)
  98.     return 0;
  99.   
  100.   a = getc(s->f)<<24;
  101.   a |= getc(s->f)<<16;
  102.   a |= getc(s->f)<<8;
  103.   b = getc(s->f);
  104.   if (b==EOF) {
  105.     s->eof = 1;
  106.     return 0;
  107.   }
  108.   return a|b;
  109. }
  110. /* ---------- functions ---------- */
  111. /* AuInit() */
  112. /* Init audio i/o streams. */
  113. void AuInit (
  114.   int debugLevel) /* in: debug level */
  115. /*     0=off  1=basic  2=full */
  116. {
  117.   AUdebugLevel = debugLevel;
  118.   if (AUdebugLevel) {
  119.     printf("AuInit: debugLevel=%dn",AUdebugLevel);
  120.   }
  121. }
  122. /* AuOpenRead() */
  123. /* Open audio stream for reading. */
  124. AuStream *AuOpenRead (
  125.   char *streamName, /* in: stream name, "-" for stdin */
  126.   int *numChannel, /* out: number of channels */
  127.   float *fSample, /* out: sampling frequency [Hz] */
  128.   long *numSample) /* out: number of samples in stream */
  129. /*      or -1 if not available */
  130. /* returns: */
  131. /*  audio stream (handle) */
  132. /*  or NULL if error */
  133. {
  134.   AuStream *s;
  135.   long h,ofs,dsize,formc,srate,nchan;
  136.   if (AUdebugLevel)
  137.     printf("AuOpenRead: fileName="%s"n",streamName);
  138.   if ((s=(AuStream*)malloc(sizeof(AuStream)))==NULL)
  139.     CommonExit(-1,"AuOpenRead: Can not allocate memory");
  140.   s->currentSample = 0;
  141.   s->eof = 0;
  142.   s->write = 0;
  143.   if (streamName[0]=='-' && streamName[1]==0)
  144.     s->f = stdin;
  145.   else
  146.     s->f = fopen(streamName,"rb");
  147.   if (s->f==NULL) {
  148.     CommonWarning("AuOpenRead: Can not open "%s"",streamName);
  149.  FREE(s);
  150.     return NULL;
  151.   }
  152.   h = getint(s);
  153.   if (h!=0x2e736e64) { /* magic string: .snd */
  154.     CommonWarning("AuOpenRead: Wrong magic string in "%s"",streamName);
  155.  FREE(s);
  156.     return NULL;
  157.   }
  158.   ofs = getint(s);
  159.   dsize = getint(s);
  160.   formc = getint(s); /* 3 = 16 bit uniform 2compl */
  161.   srate = getint(s);
  162.   nchan = getint(s);
  163.   for (h=24; h<ofs; h++)
  164.     if (getc(s->f)==EOF)
  165.       s->eof = 1;
  166.   if (s->eof || nchan<1 || formc!=3) {
  167.     CommonWarning("AuOpenRead: Unsupported audio format in "%s"",streamName);
  168.  FREE(s);
  169.     return NULL;
  170.   }
  171.   *numChannel = nchan;
  172.   *fSample = srate;
  173.   *numSample = (dsize<0)?-1:dsize/2;
  174.   if (AUdebugLevel)
  175.     printf("AuOpenRead: numChannel=%d  fSample=%.1f numSample=%ldn",
  176.    *numChannel,*fSample,*numSample);
  177.   return s;
  178. }
  179. /* AuOpenWrite() */
  180. /* Open audio stream for writing. */
  181. AuStream *AuOpenWrite (
  182.   char *streamName, /* in: stream name, "-" for stdout */
  183.   int numChannel, /* in: number of channels */
  184.   float fSample) /* in: sampling frequency [Hz] */
  185. /* returns: */
  186. /*  audio stream (handle) */
  187. /*  or NULL if error */
  188. {
  189.   AuStream *s;
  190.   if (AUdebugLevel) {
  191.     printf("AuOpenWrite: fileName="%s"n",streamName);
  192.     printf("AuOpenWrite: numChannel=%d  fSample=%.1fn",
  193.    numChannel,fSample);
  194.   }
  195.   if ((s=(AuStream*)malloc(sizeof(AuStream)))==NULL)
  196.     CommonExit(-1,"AuOpenWrite: Can not allocate memory");
  197.   s->currentSample = 0;
  198.   s->eof = 0;
  199.   s->write = 1;
  200.   if (streamName[0]=='-' && streamName[1]==0)
  201.     s->f = stdout;
  202.   else
  203.     s->f = fopen(streamName,"wb");
  204.   if (s->f==NULL) {
  205.     CommonWarning("AuOpenWrite: Can not open "%s"",streamName);
  206.  FREE(s);
  207.     return NULL;
  208.   }
  209.   putint(0x2e736e64,s); /* magic string: .snd */
  210.   putint(28,s); /* header size */
  211.   putint(-1,s); /* -1 = data size unknown */
  212.   putint(3,s); /* 3 = 16 bit uniform 2compl */
  213.   putint((int)(fSample+.5),s);
  214.   putint(numChannel,s);
  215.   putint(0,s); /* info string */
  216.   if (s->eof) {
  217.     CommonWarning("AuOpenWrite: Can not write to "%s"",streamName);
  218.  FREE(s);
  219.     return NULL;
  220.   }
  221.   return s;
  222. }
  223. /* AuReadData() */
  224. /* Read data from audio stream. */
  225. long AuReadData (
  226.   AuStream *stream, /* in: audio stream (handle) */
  227.   short *data, /* out: data[] */
  228.   long numSample) /* in: number of samples to be read */
  229. /* returns: */
  230. /*  number of samples read */
  231. {
  232.   long i;
  233.   if (AUdebugLevel > 1)
  234.     printf("AuReadData: numSample=%ldn",numSample);
  235.   if (stream->write)
  236.     CommonExit(1,"AuReadData: stream not in read mode");
  237.   i = 0;
  238.   while (!stream->eof && i<numSample)
  239.     data[i++] = getshort(stream);
  240.   stream->currentSample += i;
  241.   return i;
  242. }
  243. /* AuWriteData() */
  244. /* Write data to audio stream. */
  245. void AuWriteData (
  246.   AuStream *stream, /* in: audio stream (handle) */
  247.   short *data, /* in: data[] */
  248.   long numSample) /* in: number of samples to be written */
  249. {
  250.   long i;
  251.   if (AUdebugLevel > 1)
  252.     printf("AuWriteData: numSample=%ldn",numSample);
  253.   if (!stream->write)
  254.     CommonExit(1,"AuWriteData: audio file not in write mode");
  255.   for (i=0; i<numSample; i++)
  256.     putshort(data[i],stream);
  257.   stream->currentSample += numSample;
  258.   if (stream->eof)
  259.     CommonWarning("AuWriteDate: Can not write to au stream");
  260. }
  261. /* AuClose() */
  262. /* Close audio stream.*/
  263. void AuClose (
  264.   AuStream *stream) /* in: audio stream (handle) */
  265. {
  266.   if (AUdebugLevel)
  267.     printf("AuClose: currentSample=%ldn",stream->currentSample);
  268.   if (stream->f!=stdin && stream->f!=stdout)
  269.     fclose(stream->f);
  270. free(stream);
  271. }
  272. /* end of austream.c */