// makeWav.cc  P. Conrad   CISC181 Fall 2005
// Write a WAV file that plays a sine wave; 
// Specify duration, freq, filename on the command line

#include <iostream>
using namespace std;


// the following header files are needed for writing binary files

// they provide the functions open() and write(), as well as O_CREAT,
// O_RDWR

#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>

#include <stdio.h> // for perror

#include <cmath>   // for sin, floor
#include <cstdlib> // for atoi, atof

bool systemIsLittleEndian();
int convertToLittleEndian(int x);
signed short convertShortToLittleEndian(signed short x);



void writeWAVheader(int numSamples, int samplesPerSecond, int fd)
{
  
  char riffHeader[] = "RIFF";
  write(fd, riffHeader, 4);
  
  int lengthField = convertToLittleEndian((numSamples * 2) + 36);
  write(fd, &lengthField, 4);
  
  char WAVEfmt_Headers[] = "WAVEfmt ";
  write(fd, WAVEfmt_Headers, 8);
  
  int formatChunkSize = convertToLittleEndian(16);
  write(fd, &formatChunkSize, 4);
  
  signed short numChannels = 1;
  signed short numChannelsLittleEndian = 
    convertShortToLittleEndian(numChannels);
  write(fd, &numChannelsLittleEndian, 2);
  
  signed short compression = 1;
  signed short compressionLittleEndian = 
    convertShortToLittleEndian(compression);
  write(fd, &compressionLittleEndian, 2);
  
  int samplesPerSecondField = convertToLittleEndian(samplesPerSecond);
  write(fd, &samplesPerSecondField, 4);
  
  int bytesPerSecondField = convertToLittleEndian(samplesPerSecond * 2);
  write(fd, &bytesPerSecondField, 4);
  
  signed short bytesPerSample = 2;
  signed short bytesPerSampleLittleEndian = 
    convertShortToLittleEndian(bytesPerSample);
  write(fd, &bytesPerSampleLittleEndian, 2);
  
  signed short bitsPerSample = 16;
  signed short bitsPerSampleLittleEndian = 
    convertShortToLittleEndian(bitsPerSample);
  write(fd, &bitsPerSampleLittleEndian, 2);
  
  
  char dataHeader[] = "data";
  write(fd, dataHeader, 4);
  
  int dataLength = convertToLittleEndian(numSamples * 2);
  write(fd, &dataLength, 4);
}


int main(int argc, char *argv[])
{
  
  if (argc!=4)
    {
      cerr << "Usage: " << argv[0] 
	   << " numSeconds frequency filename" << endl;
      exit(1);
    }

  double duration = atof(argv[1]);
  
  if (duration <= 0 || duration > 60)
    {
      cerr << "Error: numSeconds must be between 1 and 60" << endl;
      exit(2);
    }

  const int samplesPerSecond = 11025;

  int frequency = atoi(argv[2]);

  if (frequency < 20 || frequency > (samplesPerSecond / 2))
    {
      cerr << "Error: frequency must be between 20 and 5012" << endl;
      exit(2);
    }


  int fd = open(argv[3], O_RDWR | O_CREAT );
  
  // make sure open was successful
  
  if (fd < 0)
    {
      cerr << "File " << argv[3] << " could not be opened" << endl;
      perror("open:");
      exit(1);
    }
  
  int numSamples = (int) floor(duration * samplesPerSecond);
    
  // Write the header to the file
  
  writeWAVheader(numSamples, samplesPerSecond, fd);
  
  // Write the samples to the file
  
  double t = 0;
  double sample;
  signed short sixteenBitSample;
  for (int i=0; i<numSamples; i++)
    {

      sample = sin(frequency * 2 * M_PI * t);
      sixteenBitSample = (signed short) floor(sample * 32767);
      sixteenBitSample = convertShortToLittleEndian(sixteenBitSample);
      write(fd,&sixteenBitSample,2);
      t += 1.0/11025.0;
    }

  close(fd);

}




