/******************************************************************************
 *                                                                            *
 * Copyright (c) 1999-2004 Wimba S.A., All Rights Reserved.                   *
 *                                                                            *
 * COPYRIGHT:                                                                 *
 *      This software is the property of Wimba S.A.                           *
 *      This software is redistributed under the Xiph.org variant of          *
 *      the BSD license.                                                      *
 *      Redistribution and use in source and binary forms, with or without    *
 *      modification, are permitted provided that the following conditions    *
 *      are met:                                                              *
 *      - Redistributions of source code must retain the above copyright      *
 *      notice, this list of conditions and the following disclaimer.         *
 *      - Redistributions in binary form must reproduce the above copyright   *
 *      notice, this list of conditions and the following disclaimer in the   *
 *      documentation and/or other materials provided with the distribution.  *
 *      - Neither the name of Wimba, the Xiph.org Foundation nor the names of *
 *      its contributors may be used to endorse or promote products derived   *
 *      from this software without specific prior written permission.         *
 *                                                                            *
 * WARRANTIES:                                                                *
 *      This software is made available by the authors in the hope            *
 *      that it will be useful, but without any warranty.                     *
 *      Wimba S.A. is not liable for any consequence related to the           *
 *      use of the provided software.                                         *
 *                                                                            *
 * Class: RawWriter.java                                                      *
 *                                                                            *
 * Author: Marc GIMPEL                                                        *
 *                                                                            *
 * Date: 6th January 2004                                                     *
 *                                                                            *
 ******************************************************************************/

/* $Id: AudioFileWriter.java,v 1.2 2004/10/21 16:21:57 mgimpel Exp $ */

package org.xiph.speex;

import java.io.DataOutput;
import java.io.File;
import java.io.IOException;
import java.io.OutputStream;

/**
 * Abstract Class that defines an Audio File Writer.
 * 
 * @author Marc Gimpel, Wimba S.A. (mgimpel@horizonwimba.com)
 * @version $Revision: 1.2 $
 */
public abstract class AudioFileWriter
{
  /**
   * Closes the output file.
   * @exception IOException if there was an exception closing the Audio Writer.
   */
  public abstract void close()
    throws IOException;
  
  /**
   * Open the output file. 
   * @param file - file to open.
   * @exception IOException if there was an exception opening the Audio Writer.
   */
  public abstract void open(File file)
    throws IOException;

  /**
   * Open the output file. 
   * @param filename - file to open.
   * @exception IOException if there was an exception opening the Audio Writer.
   */
  public abstract void open(String filename)
    throws IOException;

  /**
   * Writes the header pages that start the Ogg Speex file. 
   * Prepares file for data to be written.
   * @param comment description to be included in the header.
   * @exception IOException
   */
  public abstract void writeHeader(String comment)
    throws IOException;
  
  /**
   * Writes a packet of audio. 
   * @param data audio data
   * @param offset the offset from which to start reading the data.
   * @param len the length of data to read.
   * @exception IOException
   */
  public abstract void writePacket(byte[] data, int offset, int len)
    throws IOException;

  /**
   * Writes an Ogg Page Header to the given byte array.
   * @param buf     the buffer to write to.
   * @param offset  the from which to start writing.
   * @param headerType the header type flag
   *          (0=normal, 2=bos: beginning of stream, 4=eos: end of stream).
   * @param granulepos the absolute granule position.
   * @param streamSerialNumber
   * @param pageCount
   * @param packetCount
   * @param packetSizes
   * @return the amount of data written to the buffer.
   */
  public static int writeOggPageHeader(byte[] buf, int offset, int headerType,
                                       long granulepos, int streamSerialNumber,
                                       int pageCount, int packetCount,
                                       byte[] packetSizes)
  {
    writeString(buf, offset, "OggS");             //  0 -  3: capture_pattern
    buf[offset+4] = 0;                            //       4: stream_structure_version
    buf[offset+5] = (byte) headerType;            //       5: header_type_flag
    writeLong(buf, offset+6, granulepos);         //  6 - 13: absolute granule position
    writeInt(buf, offset+14, streamSerialNumber); // 14 - 17: stream serial number
    writeInt(buf, offset+18, pageCount);          // 18 - 21: page sequence no
    writeInt(buf, offset+22, 0);                  // 22 - 25: page checksum
    buf[offset+26] = (byte) packetCount;          //      26: page_segments
    System.arraycopy(packetSizes, 0,              // 27 -  x: segment_table
                     buf, offset+27, packetCount);
    return packetCount+27;
  }

  /**
   * Builds and returns an Ogg Page Header.
   * @param headerType the header type flag
   *          (0=normal, 2=bos: beginning of stream, 4=eos: end of stream).
   * @param granulepos the absolute granule position.
   * @param streamSerialNumber
   * @param pageCount
   * @param packetCount
   * @param packetSizes
   * @return an Ogg Page Header.
   */
  public static byte[] buildOggPageHeader(int headerType, long granulepos,
                                          int streamSerialNumber, int pageCount,
                                          int packetCount, byte[] packetSizes)
  {
    byte[] data = new byte[packetCount+27];
    writeOggPageHeader(data, 0, headerType, granulepos, streamSerialNumber,
                       pageCount, packetCount, packetSizes);
    return data;
  }

  /**
   * Writes a Speex Header to the given byte array.
   * @param buf     the buffer to write to.
   * @param offset  the from which to start writing.
   * @param sampleRate
   * @param mode
   * @param channels
   * @param vbr
   * @param nframes
   * @return the amount of data written to the buffer.
   */
  public static int writeSpeexHeader(byte[] buf, int offset, int sampleRate,
                                     int mode, int channels, boolean vbr,
                                     int nframes)
  {
    writeString(buf, offset, "Speex   ");    //  0 -  7: speex_string
    writeString(buf, offset+8, "speex-1.0"); //  8 - 27: speex_version
    System.arraycopy(new byte[11], 0, buf, offset+17, 11); // : speex_version (fill in up to 20 bytes)
    writeInt(buf, offset+28, 1);           // 28 - 31: speex_version_id
    writeInt(buf, offset+32, 80);          // 32 - 35: header_size
    writeInt(buf, offset+36, sampleRate);  // 36 - 39: rate
    writeInt(buf, offset+40, mode);        // 40 - 43: mode (0=NB, 1=WB, 2=UWB)
    writeInt(buf, offset+44, 4);           // 44 - 47: mode_bitstream_version
    writeInt(buf, offset+48, channels);    // 48 - 51: nb_channels
    writeInt(buf, offset+52, -1);          // 52 - 55: bitrate
    writeInt(buf, offset+56, 160 << mode); // 56 - 59: frame_size (NB=160, WB=320, UWB=640)
    writeInt(buf, offset+60, vbr?1:0);     // 60 - 63: vbr
    writeInt(buf, offset+64, nframes);     // 64 - 67: frames_per_packet
    writeInt(buf, offset+68, 0);           // 68 - 71: extra_headers
    writeInt(buf, offset+72, 0);           // 72 - 75: reserved1
    writeInt(buf, offset+76, 0);           // 76 - 79: reserved2
    return 80;
  }
  
  /**
   * Builds a Speex Header.
   * @param sampleRate
   * @param mode
   * @param channels
   * @param vbr
   * @param nframes
   * @return a Speex Header.
   */
  public static byte[] buildSpeexHeader(int sampleRate, int mode, int channels,
                                        boolean vbr, int nframes)
  {
    byte[] data = new byte[80];
    writeSpeexHeader(data, 0, sampleRate, mode, channels, vbr, nframes);
    return data;
  }

  /**
   * Writes a Speex Comment to the given byte array.
   * @param buf     the buffer to write to.
   * @param offset  the from which to start writing.
   * @param comment the comment.
   * @return the amount of data written to the buffer.
   */
  public static int writeSpeexComment(byte[] buf, int offset, String comment)
  {
    int length = comment.length();
    writeInt(buf, offset, length);       // vendor comment size
    writeString(buf, offset+4, comment); // vendor comment
    writeInt(buf, offset+length+4, 0);   // user comment list length
    return length+8;
  }

  /**
   * Builds and returns a Speex Comment.
   * @param comment the comment.
   * @return a Speex Comment.
   */
  public static byte[] buildSpeexComment(String comment)
  {
    byte[] data = new byte[comment.length()+8];
    writeSpeexComment(data, 0, comment);
    return data;
  }
  
  /**
   * Writes a Little-endian short.
   * @param out the data output to write to.
   * @param v value to write.
   * @exception IOException
   */  
  public static void writeShort(DataOutput out, short v)
    throws IOException 
  {
    out.writeByte((0xff & v));
    out.writeByte((0xff & (v >>> 8)));
  }
  
  /**
   * Writes a Little-endian int.
   * @param out the data output to write to.
   * @param v value to write.
   * @exception IOException
   */
  public static void writeInt(DataOutput out, int v)
    throws IOException 
  {
    out.writeByte(0xff & v);
    out.writeByte(0xff & (v >>>  8));
    out.writeByte(0xff & (v >>> 16));
    out.writeByte(0xff & (v >>> 24));
  }

  /**
   * Writes a Little-endian short.
   * @param os - the output stream to write to.
   * @param v - the value to write.
   * @exception IOException
   */
  public static void writeShort(OutputStream os, short v)
    throws IOException 
  {
    os.write((0xff & v));
    os.write((0xff & (v >>> 8)));
  }
  
  /**
   * Writes a Little-endian int.
   * @param os - the output stream to write to.
   * @param v - the value to write.
   * @exception IOException
   */
  public static void writeInt(OutputStream os, int v)
    throws IOException 
  {
    os.write(0xff & v);
    os.write(0xff & (v >>>  8));
    os.write(0xff & (v >>> 16));
    os.write(0xff & (v >>> 24));
  }

  /**
   * Writes a Little-endian long.
   * @param os - the output stream to write to.
   * @param v - the value to write.
   * @exception IOException
   */
  public static void writeLong(OutputStream os, long v)
    throws IOException
  {
    os.write((int)(0xff & v));
    os.write((int)(0xff & (v >>>  8)));
    os.write((int)(0xff & (v >>> 16)));
    os.write((int)(0xff & (v >>> 24)));
    os.write((int)(0xff & (v >>> 32)));
    os.write((int)(0xff & (v >>> 40)));
    os.write((int)(0xff & (v >>> 48)));
    os.write((int)(0xff & (v >>> 56)));
  }

  /**
   * Writes a Little-endian short.
   * @param data   the array into which the data should be written.
   * @param offset the offset from which to start writing in the array.
   * @param v      the value to write.
   */
  public static void writeShort(byte[] data, int offset, int v)
  {
    data[offset]   = (byte) (0xff & v);
    data[offset+1] = (byte) (0xff & (v >>>  8));
  }

  /**
   * Writes a Little-endian int.
   * @param data   the array into which the data should be written.
   * @param offset the offset from which to start writing in the array.
   * @param v      the value to write.
   */
  public static void writeInt(byte[] data, int offset, int v)
  {
    data[offset]   = (byte) (0xff & v);
    data[offset+1] = (byte) (0xff & (v >>>  8));
    data[offset+2] = (byte) (0xff & (v >>> 16));
    data[offset+3] = (byte) (0xff & (v >>> 24));
  }

  /**
   * Writes a Little-endian long.
   * @param data   the array into which the data should be written.
   * @param offset the offset from which to start writing in the array.
   * @param v      the value to write.
   */
  public static void writeLong(byte[] data, int offset, long v)
  {
    data[offset]   = (byte) (0xff & v);
    data[offset+1] = (byte) (0xff & (v >>>  8));
    data[offset+2] = (byte) (0xff & (v >>> 16));
    data[offset+3] = (byte) (0xff & (v >>> 24));
    data[offset+4] = (byte) (0xff & (v >>> 32));
    data[offset+5] = (byte) (0xff & (v >>> 40));
    data[offset+6] = (byte) (0xff & (v >>> 48));
    data[offset+7] = (byte) (0xff & (v >>> 56));
  }

  /**
   * Writes a String.
   * @param data   the array into which the data should be written.
   * @param offset the offset from which to start writing in the array.
   * @param v      the value to write.
   */
  public static void writeString(byte[] data, int offset, String v)
  {
    byte[] str = v.getBytes();
    System.arraycopy(str, 0, data, offset, str.length);
  }
}
