speexdec.c
上传用户:wstnjxml
上传日期:2014-04-03
资源大小:7248k
文件大小:23k
源码类别:

Windows CE

开发平台:

C/C++

  1. /* Copyright (C) 2002-2003 Jean-Marc Valin 
  2.    File: speexdec.c
  3.    Redistribution and use in source and binary forms, with or without
  4.    modification, are permitted provided that the following conditions
  5.    are met:
  6.    
  7.    - Redistributions of source code must retain the above copyright
  8.    notice, this list of conditions and the following disclaimer.
  9.    
  10.    - Redistributions in binary form must reproduce the above copyright
  11.    notice, this list of conditions and the following disclaimer in the
  12.    documentation and/or other materials provided with the distribution.
  13.    
  14.    - Neither the name of the Xiph.org Foundation nor the names of its
  15.    contributors may be used to endorse or promote products derived from
  16.    this software without specific prior written permission.
  17.    
  18.    THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
  19.    ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
  20.    LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
  21.    A PARTICULAR PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR
  22.    CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
  23.    EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
  24.    PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
  25.    PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
  26.    LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
  27.    NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  28.    SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  29. */
  30. #ifdef HAVE_CONFIG_H
  31. # include "config.h"
  32. #endif
  33. #include <stdio.h>
  34. #if !defined WIN32 && !defined _WIN32
  35. #include <unistd.h>
  36. #include <getopt.h>
  37. #endif
  38. #ifndef HAVE_GETOPT_LONG
  39. #include "getopt_win.h"
  40. #endif
  41. #include <stdlib.h>
  42. #include <string.h>
  43. #include <speex/speex.h>
  44. #include <ogg/ogg.h>
  45. #if defined WIN32 || defined _WIN32
  46. #include <windows.h>
  47. #include "getopt_win.h"
  48. #include "wave_out.h"
  49. /* We need the following two to set stdout to binary */
  50. #include <io.h>
  51. #include <fcntl.h>
  52. #endif
  53. #include <math.h>
  54. #ifdef __MINGW32__
  55. #include "wave_out.c"
  56. #endif
  57. #ifdef HAVE_SYS_SOUNDCARD_H
  58. #include <sys/soundcard.h>
  59. #include <sys/types.h>
  60. #include <sys/stat.h>
  61. #include <fcntl.h>
  62. #include <sys/ioctl.h>
  63. #elif defined HAVE_SYS_AUDIOIO_H
  64. #include <sys/types.h>
  65. #include <fcntl.h>
  66. #include <sys/ioctl.h>
  67. #include <sys/audioio.h>
  68. #ifndef AUDIO_ENCODING_SLINEAR
  69. #define AUDIO_ENCODING_SLINEAR AUDIO_ENCODING_LINEAR /* Solaris */
  70. #endif
  71. #endif
  72. #include <string.h>
  73. #include "wav_io.h"
  74. #include <speex/speex_header.h>
  75. #include <speex/speex_stereo.h>
  76. #include <speex/speex_callbacks.h>
  77. #include "wav_io.h"
  78. #define MAX_FRAME_SIZE 2000
  79. #define readint(buf, base) (((buf[base+3]<<24)&0xff000000)| 
  80.                            ((buf[base+2]<<16)&0xff0000)| 
  81.                            ((buf[base+1]<<8)&0xff00)| 
  82.                    (buf[base]&0xff))
  83. static void print_comments(char *comments, int length)
  84. {
  85.    char *c=comments;
  86.    int len, i, nb_fields;
  87.    char *end;
  88.    
  89.    if (length<8)
  90.    {
  91.       fprintf (stderr, "Invalid/corrupted commentsn");
  92.       return;
  93.    }
  94.    end = c+length;
  95.    len=readint(c, 0);
  96.    c+=4;
  97.    if (c+len>end)
  98.    {
  99.       fprintf (stderr, "Invalid/corrupted commentsn");
  100.       return;
  101.    }
  102.    fwrite(c, 1, len, stderr);
  103.    c+=len;
  104.    fprintf (stderr, "n");
  105.    if (c+4>end)
  106.    {
  107.       fprintf (stderr, "Invalid/corrupted commentsn");
  108.       return;
  109.    }
  110.    nb_fields=readint(c, 0);
  111.    c+=4;
  112.    for (i=0;i<nb_fields;i++)
  113.    {
  114.       if (c+4>end)
  115.       {
  116.          fprintf (stderr, "Invalid/corrupted commentsn");
  117.          return;
  118.       }
  119.       len=readint(c, 0);
  120.       c+=4;
  121.       if (c+len>end)
  122.       {
  123.          fprintf (stderr, "Invalid/corrupted commentsn");
  124.          return;
  125.       }
  126.       fwrite(c, 1, len, stderr);
  127.       c+=len;
  128.       fprintf (stderr, "n");
  129.    }
  130. }
  131. FILE *out_file_open(char *outFile, int rate, int *channels)
  132. {
  133.    FILE *fout=NULL;
  134.    /*Open output file*/
  135.    if (strlen(outFile)==0)
  136.    {
  137. #if defined HAVE_SYS_SOUNDCARD_H
  138.       int audio_fd, format, stereo;
  139.       audio_fd=open("/dev/dsp", O_WRONLY);
  140.       if (audio_fd<0)
  141.       {
  142.          perror("Cannot open /dev/dsp");
  143.          exit(1);         
  144.       }
  145.       format=AFMT_S16_NE;
  146.       if (ioctl(audio_fd, SNDCTL_DSP_SETFMT, &format)==-1)
  147.       {
  148.          perror("SNDCTL_DSP_SETFMT");
  149.          close(audio_fd);
  150.          exit(1);
  151.       }
  152.       stereo=0;
  153.       if (*channels==2)
  154.          stereo=1;
  155.       if (ioctl(audio_fd, SNDCTL_DSP_STEREO, &stereo)==-1)
  156.       {
  157.          perror("SNDCTL_DSP_STEREO");
  158.          close(audio_fd);
  159.          exit(1);
  160.       }
  161.       if (stereo!=0)
  162.       {
  163.          if (*channels==1)
  164.             fprintf (stderr, "Cannot set mono mode, will decode in stereon");
  165.          *channels=2;
  166.       }
  167.       if (ioctl(audio_fd, SNDCTL_DSP_SPEED, &rate)==-1)
  168.       {
  169.          perror("SNDCTL_DSP_SPEED");
  170.          close(audio_fd);
  171.          exit(1);
  172.       }
  173.       fout = fdopen(audio_fd, "w");
  174. #elif defined HAVE_SYS_AUDIOIO_H
  175.       audio_info_t info;
  176.       int audio_fd;
  177.       
  178.       audio_fd = open("/dev/audio", O_WRONLY);
  179.       if (audio_fd<0)
  180.       {
  181.          perror("Cannot open /dev/audio");
  182.          exit(1);
  183.       }
  184.       AUDIO_INITINFO(&info);
  185. #ifdef AUMODE_PLAY    /* NetBSD/OpenBSD */
  186.       info.mode = AUMODE_PLAY;
  187. #endif
  188.       info.play.encoding = AUDIO_ENCODING_SLINEAR;
  189.       info.play.precision = 16;
  190.       info.play.sample_rate = rate;
  191.       info.play.channels = *channels;
  192.       
  193.       if (ioctl(audio_fd, AUDIO_SETINFO, &info) < 0)
  194.       {
  195.          perror ("AUDIO_SETINFO");
  196.          exit(1);
  197.       }
  198.       fout = fdopen(audio_fd, "w");
  199. #elif defined WIN32 || defined _WIN32
  200.       {
  201.          unsigned int speex_channels = *channels;
  202.          if (Set_WIN_Params (INVALID_FILEDESC, rate, SAMPLE_SIZE, speex_channels))
  203.          {
  204.             fprintf (stderr, "Can't access %sn", "WAVE OUT");
  205.             exit(1);
  206.          }
  207.       }
  208. #else
  209.       fprintf (stderr, "No soundcard supportn");
  210.       exit(1);
  211. #endif
  212.    } else {
  213.       if (strcmp(outFile,"-")==0)
  214.       {
  215. #if defined WIN32 || defined _WIN32
  216.          _setmode(_fileno(stdout), _O_BINARY);
  217. #endif
  218.          fout=stdout;
  219.       }
  220.       else 
  221.       {
  222.          fout = fopen(outFile, "wb");
  223.          if (!fout)
  224.          {
  225.             perror(outFile);
  226.             exit(1);
  227.          }
  228.          if (strcmp(outFile+strlen(outFile)-4,".wav")==0 || strcmp(outFile+strlen(outFile)-4,".WAV")==0)
  229.             write_wav_header(fout, rate, *channels, 0, 0);
  230.       }
  231.    }
  232.    return fout;
  233. }
  234. void usage()
  235. {
  236.    printf ("Usage: speexdec [options] input_file.spx [output_file]n");
  237.    printf ("n");
  238.    printf ("Decodes a Speex file and produce a WAV file or raw filen");
  239.    printf ("n");
  240.    printf ("input_file can be:n");
  241.    printf ("  filename.spx         regular Speex filen");
  242.    printf ("  -                    stdinn");
  243.    printf ("n");  
  244.    printf ("output_file can be:n");
  245.    printf ("  filename.wav         Wav filen");
  246.    printf ("  filename.*           Raw PCM file (any extension other that .wav)n");
  247.    printf ("  -                    stdoutn");
  248.    printf ("  (nothing)            Will be played to soundcardn");
  249.    printf ("n");  
  250.    printf ("Options:n");
  251.    printf (" --enh                 Enable perceptual enhancement (default)n");
  252.    printf (" --no-enh              Disable perceptual enhancementn");
  253.    printf (" --force-nb            Force decoding in narrowbandn");
  254.    printf (" --force-wb            Force decoding in widebandn");
  255.    printf (" --force-uwb           Force decoding in ultra-widebandn");
  256.    printf (" --mono                Force decoding in monon");
  257.    printf (" --stereo              Force decoding in stereon");
  258.    printf (" --rate n              Force decoding at sampling rate n Hzn");
  259.    printf (" --packet-loss n       Simulate n %% random packet lossn");
  260.    printf (" -V                    Verbose mode (show bit-rate)n"); 
  261.    printf (" -h, --help            This helpn");
  262.    printf (" -v, --version         Version informationn");
  263.    printf (" --pf                  Deprecated, use --enh insteadn");
  264.    printf (" --no-pf               Deprecated, use --no-enh insteadn");
  265.    printf ("n");
  266.    printf ("More information is available from the Speex site: http://www.speex.orgn");
  267.    printf ("n");
  268.    printf ("Please report bugs to the mailing list `speex-dev@xiph.org'.n");
  269. }
  270. void version()
  271. {
  272.    printf ("speexdec (Speex decoder) version " SPEEX_VERSION " (compiled " __DATE__ ")n");
  273.    printf ("Copyright (C) 2002-2003 Jean-Marc Valinn");
  274. }
  275. void version_short()
  276. {
  277.    printf ("speexdec version " SPEEX_VERSION "n");
  278.    printf ("Copyright (C) 2002-2003 Jean-Marc Valinn");
  279. }
  280. static void *process_header(ogg_packet *op, int enh_enabled, int *frame_size, int *rate, int *nframes, int forceMode, int *channels, SpeexStereoState *stereo, int *extra_headers, int quiet)
  281. {
  282.    void *st;
  283.    const SpeexMode *mode;
  284.    SpeexHeader *header;
  285.    int modeID;
  286.    SpeexCallback callback;
  287.       
  288.    header = speex_packet_to_header((char*)op->packet, op->bytes);
  289.    if (!header)
  290.    {
  291.       fprintf (stderr, "Cannot read headern");
  292.       return NULL;
  293.    }
  294.    if (header->mode >= SPEEX_NB_MODES)
  295.    {
  296.       fprintf (stderr, "Mode number %d does not (yet/any longer) exist in this versionn", 
  297.                header->mode);
  298.       return NULL;
  299.    }
  300.       
  301.    modeID = header->mode;
  302.    if (forceMode!=-1)
  303.       modeID = forceMode;
  304.    mode = speex_lib_get_mode (modeID);
  305.    
  306.    if (header->speex_version_id > 1)
  307.    {
  308.       fprintf (stderr, "This file was encoded with Speex bit-stream version %d, which I don't know how to decoden", header->speex_version_id);
  309.       return NULL;
  310.    }
  311.    if (mode->bitstream_version < header->mode_bitstream_version)
  312.    {
  313.       fprintf (stderr, "The file was encoded with a newer version of Speex. You need to upgrade in order to play it.n");
  314.       return NULL;
  315.    }
  316.    if (mode->bitstream_version > header->mode_bitstream_version) 
  317.    {
  318.       fprintf (stderr, "The file was encoded with an older version of Speex. You would need to downgrade the version in order to play it.n");
  319.       return NULL;
  320.    }
  321.    
  322.    st = speex_decoder_init(mode);
  323.    if (!st)
  324.    {
  325.       fprintf (stderr, "Decoder initialization failed.n");
  326.       return NULL;
  327.    }
  328.    speex_decoder_ctl(st, SPEEX_SET_ENH, &enh_enabled);
  329.    speex_decoder_ctl(st, SPEEX_GET_FRAME_SIZE, frame_size);
  330.    if (!(*channels==1))
  331.    {
  332.       callback.callback_id = SPEEX_INBAND_STEREO;
  333.       callback.func = speex_std_stereo_request_handler;
  334.       callback.data = stereo;
  335.       speex_decoder_ctl(st, SPEEX_SET_HANDLER, &callback);
  336.    }
  337.    if (!*rate)
  338.       *rate = header->rate;
  339.    /* Adjust rate if --force-* options are used */
  340.    if (forceMode!=-1)
  341.    {
  342.       if (header->mode < forceMode)
  343.          *rate <<= (forceMode - header->mode);
  344.       if (header->mode > forceMode)
  345.          *rate >>= (header->mode - forceMode);
  346.    }
  347.    speex_decoder_ctl(st, SPEEX_SET_SAMPLING_RATE, rate);
  348.    *nframes = header->frames_per_packet;
  349.    if (*channels==-1)
  350.       *channels = header->nb_channels;
  351.    
  352.    if (!quiet)
  353.    {
  354.       fprintf (stderr, "Decoding %d Hz audio using %s mode", 
  355.                *rate, mode->modeName);
  356.       if (*channels==1)
  357.          fprintf (stderr, " (mono");
  358.       else
  359.          fprintf (stderr, " (stereo");
  360.       
  361.       if (header->vbr)
  362.          fprintf (stderr, ", VBR)n");
  363.       else
  364.          fprintf(stderr, ")n");
  365.       /*fprintf (stderr, "Decoding %d Hz audio at %d bps using %s moden", 
  366.        *rate, mode->bitrate, mode->modeName);*/
  367.    }
  368.    *extra_headers = header->extra_headers;
  369.    free(header);
  370.    return st;
  371. }
  372. int main(int argc, char **argv)
  373. {
  374.    int c;
  375.    int option_index = 0;
  376.    char *inFile, *outFile;
  377.    FILE *fin, *fout=NULL;
  378.    short out[MAX_FRAME_SIZE];
  379.    short output[MAX_FRAME_SIZE];
  380.    int frame_size=0;
  381.    void *st=NULL;
  382.    SpeexBits bits;
  383.    int packet_count=0;
  384.    int stream_init = 0;
  385.    int quiet = 0;
  386.    ogg_int64_t page_granule=0, last_granule=0;
  387.    int skip_samples=0, page_nb_packets;
  388.    struct option long_options[] =
  389.    {
  390.       {"help", no_argument, NULL, 0},
  391.       {"quiet", no_argument, NULL, 0},
  392.       {"version", no_argument, NULL, 0},
  393.       {"version-short", no_argument, NULL, 0},
  394.       {"enh", no_argument, NULL, 0},
  395.       {"no-enh", no_argument, NULL, 0},
  396.       {"pf", no_argument, NULL, 0},
  397.       {"no-pf", no_argument, NULL, 0},
  398.       {"force-nb", no_argument, NULL, 0},
  399.       {"force-wb", no_argument, NULL, 0},
  400.       {"force-uwb", no_argument, NULL, 0},
  401.       {"rate", required_argument, NULL, 0},
  402.       {"mono", no_argument, NULL, 0},
  403.       {"stereo", no_argument, NULL, 0},
  404.       {"packet-loss", required_argument, NULL, 0},
  405.       {0, 0, 0, 0}
  406.    };
  407.    ogg_sync_state oy;
  408.    ogg_page       og;
  409.    ogg_packet     op;
  410.    ogg_stream_state os;
  411.    int enh_enabled;
  412.    int nframes=2;
  413.    int print_bitrate=0;
  414.    int close_in=0;
  415.    int eos=0;
  416.    int forceMode=-1;
  417.    int audio_size=0;
  418.    float loss_percent=-1;
  419.    SpeexStereoState stereo = SPEEX_STEREO_STATE_INIT;
  420.    int channels=-1;
  421.    int rate=0;
  422.    int extra_headers;
  423.    int wav_format=0;
  424.    enh_enabled = 1;
  425.    /*Process options*/
  426.    while(1)
  427.    {
  428.       c = getopt_long (argc, argv, "hvV",
  429.                        long_options, &option_index);
  430.       if (c==-1)
  431.          break;
  432.       
  433.       switch(c)
  434.       {
  435.       case 0:
  436.          if (strcmp(long_options[option_index].name,"help")==0)
  437.          {
  438.             usage();
  439.             exit(0);
  440.          } else if (strcmp(long_options[option_index].name,"quiet")==0)
  441.          {
  442.             quiet = 1;
  443.          } else if (strcmp(long_options[option_index].name,"version")==0)
  444.          {
  445.             version();
  446.             exit(0);
  447.          } else if (strcmp(long_options[option_index].name,"version-short")==0)
  448.          {
  449.             version_short();
  450.             exit(0);
  451.          } else if (strcmp(long_options[option_index].name,"enh")==0)
  452.          {
  453.             enh_enabled=1;
  454.          } else if (strcmp(long_options[option_index].name,"no-enh")==0)
  455.          {
  456.             enh_enabled=0;
  457.          } else if (strcmp(long_options[option_index].name,"pf")==0)
  458.          {
  459.             fprintf (stderr, "--pf is deprecated, use --enh insteadn");
  460.             enh_enabled=1;
  461.          } else if (strcmp(long_options[option_index].name,"no-pf")==0)
  462.          {
  463.             fprintf (stderr, "--no-pf is deprecated, use --no-enh insteadn");
  464.             enh_enabled=0;
  465.          } else if (strcmp(long_options[option_index].name,"force-nb")==0)
  466.          {
  467.             forceMode=0;
  468.          } else if (strcmp(long_options[option_index].name,"force-wb")==0)
  469.          {
  470.             forceMode=1;
  471.          } else if (strcmp(long_options[option_index].name,"force-uwb")==0)
  472.          {
  473.             forceMode=2;
  474.          } else if (strcmp(long_options[option_index].name,"mono")==0)
  475.          {
  476.             channels=1;
  477.          } else if (strcmp(long_options[option_index].name,"stereo")==0)
  478.          {
  479.             channels=2;
  480.          } else if (strcmp(long_options[option_index].name,"rate")==0)
  481.          {
  482.             rate=atoi (optarg);
  483.          } else if (strcmp(long_options[option_index].name,"packet-loss")==0)
  484.          {
  485.             loss_percent = atof(optarg);
  486.          }
  487.          break;
  488.       case 'h':
  489.          usage();
  490.          exit(0);
  491.          break;
  492.       case 'v':
  493.          version();
  494.          exit(0);
  495.          break;
  496.       case 'V':
  497.          print_bitrate=1;
  498.          break;
  499.       case '?':
  500.          usage();
  501.          exit(1);
  502.          break;
  503.       }
  504.    }
  505.    if (argc-optind!=2 && argc-optind!=1)
  506.    {
  507.       usage();
  508.       exit(1);
  509.    }
  510.    inFile=argv[optind];
  511.    if (argc-optind==2)
  512.       outFile=argv[optind+1];
  513.    else
  514.       outFile = "";
  515.    wav_format = strlen(outFile)>=4 && (
  516.                                        strcmp(outFile+strlen(outFile)-4,".wav")==0
  517.                                        || strcmp(outFile+strlen(outFile)-4,".WAV")==0);
  518.    /*Open input file*/
  519.    if (strcmp(inFile, "-")==0)
  520.    {
  521. #if defined WIN32 || defined _WIN32
  522.       _setmode(_fileno(stdin), _O_BINARY);
  523. #endif
  524.       fin=stdin;
  525.    }
  526.    else 
  527.    {
  528.       fin = fopen(inFile, "rb");
  529.       if (!fin)
  530.       {
  531.          perror(inFile);
  532.          exit(1);
  533.       }
  534.       close_in=1;
  535.    }
  536.    /*Init Ogg data struct*/
  537.    ogg_sync_init(&oy);
  538.    
  539.    speex_bits_init(&bits);
  540.    /*Main decoding loop*/
  541.    while (1)
  542.    {
  543.       char *data;
  544.       int i, j, nb_read;
  545.       /*Get the ogg buffer for writing*/
  546.       data = ogg_sync_buffer(&oy, 200);
  547.       /*Read bitstream from input file*/
  548.       nb_read = fread(data, sizeof(char), 200, fin);      
  549.       ogg_sync_wrote(&oy, nb_read);
  550.       /*Loop for all complete pages we got (most likely only one)*/
  551.       while (ogg_sync_pageout(&oy, &og)==1)
  552.       {
  553.          int packet_no;
  554.          if (stream_init == 0) {
  555.             ogg_stream_init(&os, ogg_page_serialno(&og));
  556.             stream_init = 1;
  557.          }
  558.          /*Add page to the bitstream*/
  559.          ogg_stream_pagein(&os, &og);
  560.          page_granule = ogg_page_granulepos(&og);
  561.          page_nb_packets = ogg_page_packets(&og);
  562.          if (page_granule>0 && frame_size)
  563.          {
  564.             skip_samples = page_nb_packets*frame_size*nframes - (page_granule-last_granule);
  565.             if (ogg_page_eos(&og))
  566.                skip_samples = -skip_samples;
  567.             /*else if (!ogg_page_bos(&og))
  568.                skip_samples = 0;*/
  569.          } else
  570.          {
  571.             skip_samples = 0;
  572.          }
  573.          /*printf ("page granulepos: %d %d %dn", skip_samples, page_nb_packets, (int)page_granule);*/
  574.          last_granule = page_granule;
  575.          /*Extract all available packets*/
  576.          packet_no=0;
  577.          while (!eos && ogg_stream_packetout(&os, &op)==1)
  578.          {
  579.             /*If first packet, process as Speex header*/
  580.             if (packet_count==0)
  581.             {
  582.                st = process_header(&op, enh_enabled, &frame_size, &rate, &nframes, forceMode, &channels, &stereo, &extra_headers, quiet);
  583.                if (!nframes)
  584.                   nframes=1;
  585.                if (!st)
  586.                   exit(1);
  587.                fout = out_file_open(outFile, rate, &channels);
  588.             } else if (packet_count==1)
  589.             {
  590.                if (!quiet)
  591.                   print_comments((char*)op.packet, op.bytes);
  592.             } else if (packet_count<=1+extra_headers)
  593.             {
  594.                /* Ignore extra headers */
  595.             } else {
  596.                int lost=0;
  597.                packet_no++;
  598.                if (loss_percent>0 && 100*((float)rand())/RAND_MAX<loss_percent)
  599.                   lost=1;
  600.                /*End of stream condition*/
  601.                if (op.e_o_s)
  602.                   eos=1;
  603.                /*Copy Ogg packet to Speex bitstream*/
  604.                speex_bits_read_from(&bits, (char*)op.packet, op.bytes);
  605.                for (j=0;j!=nframes;j++)
  606.                {
  607.                   int ret;
  608.                   /*Decode frame*/
  609.                   if (!lost)
  610.                      ret = speex_decode_int(st, &bits, output);
  611.                   else
  612.                      ret = speex_decode_int(st, NULL, output);
  613.                   /*for (i=0;i<frame_size*channels;i++)
  614.                     printf ("%dn", (int)output[i]);*/
  615.                   if (ret==-1)
  616.                      break;
  617.                   if (ret==-2)
  618.                   {
  619.                      fprintf (stderr, "Decoding error: corrupted stream?n");
  620.                      break;
  621.                   }
  622.                   if (speex_bits_remaining(&bits)<0)
  623.                   {
  624.                      fprintf (stderr, "Decoding overflow: corrupted stream?n");
  625.                      break;
  626.                   }
  627.                   if (channels==2)
  628.                      speex_decode_stereo_int(output, frame_size, &stereo);
  629.                   if (print_bitrate) {
  630.                      int tmp;
  631.                      char ch=13;
  632.                      speex_decoder_ctl(st, SPEEX_GET_BITRATE, &tmp);
  633.                      fputc (ch, stderr);
  634.                      fprintf (stderr, "Bitrate is use: %d bps     ", tmp);
  635.                   }
  636.                   /*Convert to short and save to output file*/
  637.   if (strlen(outFile)!=0)
  638.                   {
  639.                      for (i=0;i<frame_size*channels;i++)
  640.                         out[i]=le_short(output[i]);
  641.   } else {
  642.                      for (i=0;i<frame_size*channels;i++)
  643.                         out[i]=output[i];
  644.   }
  645.                   {
  646.                      int frame_offset = 0;
  647.                      int new_frame_size = frame_size;
  648.                      /*printf ("packet %d %dn", packet_no, skip_samples);*/
  649.                      if (packet_no == 1 && j==0 && skip_samples > 0)
  650.                      {
  651.                         /*printf ("chopping first packetn");*/
  652.                         new_frame_size -= skip_samples;
  653.                         frame_offset = skip_samples;
  654.                      }
  655.                      if (packet_no == page_nb_packets && skip_samples < 0)
  656.                      {
  657.                         int packet_length = nframes*frame_size+skip_samples;
  658.                         new_frame_size = packet_length - j*frame_size;
  659.                         if (new_frame_size<0)
  660.                            new_frame_size = 0;
  661.                         if (new_frame_size>frame_size)
  662.                            new_frame_size = frame_size;
  663.                         /*printf ("chopping end: %d %d %dn", new_frame_size, packet_length, packet_no);*/
  664.                      }
  665.                      if (new_frame_size)
  666.                      {  
  667. #if defined WIN32 || defined _WIN32
  668.                         if (strlen(outFile)==0)
  669.                            WIN_Play_Samples (out+frame_offset*channels, sizeof(short) * new_frame_size*channels);
  670.                         else
  671. #endif
  672.                            fwrite(out+frame_offset*channels, sizeof(short), new_frame_size*channels, fout);
  673.                   
  674.                         audio_size+=sizeof(short)*new_frame_size*channels;
  675.                      }
  676.                   }
  677.                }
  678.             }
  679.             packet_count++;
  680.          }
  681.       }
  682.       if (feof(fin))
  683.          break;
  684.    }
  685.    if (wav_format)
  686.    {
  687.       if (fseek(fout,4,SEEK_SET)==0)
  688.       {
  689.          int tmp;
  690.          tmp = le_int(audio_size+36);
  691.          fwrite(&tmp,4,1,fout);
  692.          if (fseek(fout,32,SEEK_CUR)==0)
  693.          {
  694.             tmp = le_int(audio_size);
  695.             fwrite(&tmp,4,1,fout);
  696.          } else
  697.          {
  698.             fprintf (stderr, "First seek worked, second didn'tn");
  699.          }
  700.       } else {
  701.          fprintf (stderr, "Cannot seek on wave file, size will be incorrectn");
  702.       }
  703.    }
  704.    if (st)
  705.       speex_decoder_destroy(st);
  706.    else 
  707.    {
  708.       fprintf (stderr, "This doesn't look like a Speex filen");
  709.    }
  710.    speex_bits_destroy(&bits);
  711.    if (stream_init)
  712.       ogg_stream_clear(&os);
  713.    ogg_sync_clear(&oy);
  714. #if defined WIN32 || defined _WIN32
  715.    if (strlen(outFile)==0)
  716.       WIN_Audio_close ();
  717. #endif
  718.    if (close_in)
  719.       fclose(fin);
  720.    if (fout != NULL)
  721.       fclose(fout);   
  722.    return 0;
  723. }