Blame ports/MSVC++/examples/feedseek.c

Packit c32a2d
/*
Packit c32a2d
	feedseek: test program for libmpg123, showing how to use fuzzy seeking in feeder mode
Packit c32a2d
	copyright 2008 by the mpg123 project - free software under the terms of the LGPL 2.1
Packit c32a2d
	see COPYING and AUTHORS files in distribution or http://mpg123.org	
Packit c32a2d
*/
Packit c32a2d

Packit c32a2d
#include <mpg123.h>
Packit c32a2d
#include <stdio.h>
Packit c32a2d
#include <tchar.h>
Packit c32a2d
#include <wchar.h>
Packit c32a2d

Packit c32a2d
#define INBUFF  16384 * 2 * 2
Packit c32a2d
#define WAVE_FORMAT_PCM 0x0001
Packit c32a2d
#define WAVE_FORMAT_IEEE_FLOAT 0x0003
Packit c32a2d

Packit c32a2d
FILE *out;
Packit c32a2d
size_t totaloffset, dataoffset;
Packit c32a2d
long rate;
Packit c32a2d
int channels, enc;
Packit c32a2d
unsigned short bitspersample, wavformat;
Packit c32a2d

Packit c32a2d
// write wav header
Packit c32a2d
void initwav()
Packit c32a2d
{
Packit c32a2d
	unsigned int tmp32 = 0;
Packit c32a2d
	unsigned short tmp16 = 0;
Packit c32a2d

Packit c32a2d
	fwrite("RIFF", 1, 4, out);
Packit c32a2d
	totaloffset = ftell(out);
Packit c32a2d

Packit c32a2d
	fwrite(&tmp32, 1, 4, out); // total size
Packit c32a2d
	fwrite("WAVE", 1, 4, out);
Packit c32a2d
	fwrite("fmt ", 1, 4, out);
Packit c32a2d
	tmp32 = 16;
Packit c32a2d
	fwrite(&tmp32, 1, 4, out); // format length
Packit c32a2d
	tmp16 = wavformat;
Packit c32a2d
	fwrite(&tmp16, 1, 2, out); // format
Packit c32a2d
	tmp16 = channels;
Packit c32a2d
	fwrite(&tmp16, 1, 2, out); // channels
Packit c32a2d
	tmp32 = rate;
Packit c32a2d
	fwrite(&tmp32, 1, 4, out); // sample rate
Packit c32a2d
	tmp32 = rate * bitspersample/8 * channels;
Packit c32a2d
	fwrite(&tmp32, 1, 4, out); // bytes / second
Packit c32a2d
	tmp16 = bitspersample/8 * channels; // float 16 or signed int 16
Packit c32a2d
	fwrite(&tmp16, 1, 2, out); // block align
Packit c32a2d
	tmp16 = bitspersample;
Packit c32a2d
	fwrite(&tmp16, 1, 2, out); // bits per sample
Packit c32a2d
	fwrite("data ", 1, 4, out);
Packit c32a2d
	tmp32 = 0;
Packit c32a2d
	dataoffset = ftell(out);
Packit c32a2d
	fwrite(&tmp32, 1, 4, out); // data length
Packit c32a2d
}
Packit c32a2d

Packit c32a2d
// rewrite wav header with final length infos
Packit c32a2d
void closewav()
Packit c32a2d
{
Packit c32a2d
	unsigned int tmp32 = 0;
Packit c32a2d
	unsigned short tmp16 = 0;
Packit c32a2d

Packit c32a2d
	long total = ftell(out);
Packit c32a2d
	fseek(out, totaloffset, SEEK_SET);
Packit c32a2d
	tmp32 = total - (totaloffset + 4);
Packit c32a2d
	fwrite(&tmp32, 1, 4, out);
Packit c32a2d
	fseek(out, dataoffset, SEEK_SET);
Packit c32a2d
	tmp32 = total - (dataoffset + 4);
Packit c32a2d

Packit c32a2d
	fwrite(&tmp32, 1, 4, out);
Packit c32a2d
}
Packit c32a2d

Packit c32a2d
// determine correct wav format and bits per sample
Packit c32a2d
// from mpg123 enc value
Packit c32a2d
void initwavformat()
Packit c32a2d
{
Packit c32a2d
	if(enc & MPG123_ENC_FLOAT_64)
Packit c32a2d
	{
Packit c32a2d
		bitspersample = 64;
Packit c32a2d
		wavformat = WAVE_FORMAT_IEEE_FLOAT;
Packit c32a2d
	}
Packit c32a2d
	else if(enc & MPG123_ENC_FLOAT_32)
Packit c32a2d
	{
Packit c32a2d
		bitspersample = 32;
Packit c32a2d
		wavformat = WAVE_FORMAT_IEEE_FLOAT;
Packit c32a2d
	}
Packit c32a2d
	else if(enc & MPG123_ENC_16)
Packit c32a2d
	{
Packit c32a2d
		bitspersample = 16;
Packit c32a2d
		wavformat = WAVE_FORMAT_PCM;
Packit c32a2d
	}
Packit c32a2d
	else
Packit c32a2d
	{
Packit c32a2d
		bitspersample = 8;
Packit c32a2d
		wavformat = WAVE_FORMAT_PCM;
Packit c32a2d
	}
Packit c32a2d
}
Packit c32a2d

Packit c32a2d
int _tmain(int argc, TCHAR **argv)
Packit c32a2d
{
Packit c32a2d
	unsigned char buf[INBUFF];
Packit c32a2d
	unsigned char *audio;
Packit c32a2d
	FILE *in;
Packit c32a2d
	mpg123_handle *m;
Packit c32a2d
	int ret, state;
Packit c32a2d
	size_t inc, outc;
Packit c32a2d
	off_t len, num;
Packit c32a2d
	size_t bytes;
Packit c32a2d
	off_t inoffset;
Packit c32a2d
	size_t nrates;
Packit c32a2d
	const long *rates;
Packit c32a2d
	size_t i;
Packit c32a2d
	inc = outc = 0;
Packit c32a2d
	nrates = 0;
Packit c32a2d
	rates = NULL;
Packit c32a2d

Packit c32a2d
	if(argc < 3)
Packit c32a2d
	{
Packit c32a2d
		fprintf(stderr,"Please supply in and out filenames\n");
Packit c32a2d
		return -1;
Packit c32a2d
	}
Packit c32a2d

Packit c32a2d
	mpg123_init();
Packit c32a2d

Packit c32a2d
	m = mpg123_new(NULL, &ret;;
Packit c32a2d
	if(m == NULL)
Packit c32a2d
	{
Packit c32a2d
		fprintf(stderr,"Unable to create mpg123 handle: %s\n", mpg123_plain_strerror(ret));
Packit c32a2d
		return -1;
Packit c32a2d
	}
Packit c32a2d

Packit c32a2d
	mpg123_param(m, MPG123_VERBOSE, 4, 0);
Packit c32a2d

Packit c32a2d
	ret = mpg123_param(m, MPG123_FLAGS, MPG123_FUZZY | MPG123_SEEKBUFFER | MPG123_GAPLESS, 0);
Packit c32a2d
	if(ret != MPG123_OK)
Packit c32a2d
	{
Packit c32a2d
		fprintf(stderr,"Unable to set library options: %s\n", mpg123_plain_strerror(ret));
Packit c32a2d
		return -1;
Packit c32a2d
	}
Packit c32a2d

Packit c32a2d
	// Let the seek index auto-grow and contain an entry for every frame
Packit c32a2d
	ret = mpg123_param(m, MPG123_INDEX_SIZE, -1, 0);
Packit c32a2d
	if(ret != MPG123_OK)
Packit c32a2d
	{
Packit c32a2d
		fprintf(stderr,"Unable to set index size: %s\n", mpg123_plain_strerror(ret));
Packit c32a2d
		return -1;
Packit c32a2d
	}
Packit c32a2d

Packit c32a2d
	// Use float output formats only
Packit c32a2d
	ret = mpg123_format_none(m);
Packit c32a2d
	if(ret != MPG123_OK)
Packit c32a2d
	{
Packit c32a2d
		fprintf(stderr,"Unable to disable all output formats: %s\n", mpg123_plain_strerror(ret));
Packit c32a2d
		return -1;
Packit c32a2d
	}
Packit c32a2d
	
Packit c32a2d
	mpg123_rates(&rates, &nrates);
Packit c32a2d
	for(i=0; i
Packit c32a2d
	{
Packit c32a2d
		ret = mpg123_format(m, rates[i], MPG123_MONO | MPG123_STEREO,  MPG123_ENC_FLOAT_32);
Packit c32a2d
		if(ret != MPG123_OK)
Packit c32a2d
		{
Packit c32a2d
			fprintf(stderr,"Unable to set float output formats: %s\n", mpg123_plain_strerror(ret));
Packit c32a2d
			return -1;
Packit c32a2d
		}
Packit c32a2d
	}
Packit c32a2d

Packit c32a2d
	ret = mpg123_open_feed(m);
Packit c32a2d
	if(ret != MPG123_OK)
Packit c32a2d
	{
Packit c32a2d
		fprintf(stderr,"Unable open feed: %s\n", mpg123_plain_strerror(ret));
Packit c32a2d
		return -1;
Packit c32a2d
	}
Packit c32a2d

Packit c32a2d
	in = _tfopen(argv[1], __T("rb"));
Packit c32a2d
	if(in == NULL)
Packit c32a2d
	{
Packit c32a2d
		_ftprintf(stderr,__T("Unable to open input file %s\n"), argv[1]);
Packit c32a2d
		return -1;
Packit c32a2d
	}
Packit c32a2d
	
Packit c32a2d
	out = _tfopen(argv[2], __T("wb"));
Packit c32a2d
	if(out == NULL)
Packit c32a2d
	{
Packit c32a2d
		_ftprintf(stderr,__T("Unable to open output file %s\n"), argv[2]);
Packit c32a2d
		return -1;
Packit c32a2d
	}
Packit c32a2d
	
Packit c32a2d
	while(ret = mpg123_feedseek(m, 95000, SEEK_SET, &inoffset) == MPG123_NEED_MORE)
Packit c32a2d
	{
Packit c32a2d
		len = fread(buf, sizeof(unsigned char), INBUFF, in);
Packit c32a2d
		if(len <= 0)
Packit c32a2d
			break;
Packit c32a2d
		inc += len;
Packit c32a2d

Packit c32a2d
		state = mpg123_feed(m, buf, len);
Packit c32a2d
		if(state == MPG123_ERR)
Packit c32a2d
		{
Packit c32a2d
			fprintf(stderr, "Error: %s", mpg123_strerror(m));
Packit c32a2d
			return -1; 
Packit c32a2d
		}
Packit c32a2d
	}
Packit c32a2d
	
Packit c32a2d
	fseek(in, inoffset, SEEK_SET);
Packit c32a2d
	
Packit c32a2d
	while(1)
Packit c32a2d
	{
Packit c32a2d
		len = fread(buf, sizeof(unsigned char), INBUFF, in);
Packit c32a2d
		if(len <= 0)
Packit c32a2d
			break;
Packit c32a2d
		inc += len;
Packit c32a2d
		ret = mpg123_feed(m, buf, len);
Packit c32a2d

Packit c32a2d
		while(ret != MPG123_ERR && ret != MPG123_NEED_MORE)
Packit c32a2d
		{
Packit c32a2d
			ret = mpg123_decode_frame(m, &num, &audio, &bytes);
Packit c32a2d
			if(ret == MPG123_NEW_FORMAT)
Packit c32a2d
			{
Packit c32a2d
				mpg123_getformat(m, &rate, &channels, &enc;;
Packit c32a2d
				initwavformat();
Packit c32a2d
				initwav();
Packit c32a2d
				fprintf(stderr, "New format: %li Hz, %i channels, encoding value %i\n", rate, channels, enc);
Packit c32a2d
			}
Packit c32a2d
			fwrite(audio, sizeof(unsigned char), bytes, out);
Packit c32a2d
			outc += bytes;
Packit c32a2d
		}
Packit c32a2d

Packit c32a2d
		if(ret == MPG123_ERR)
Packit c32a2d
		{
Packit c32a2d
			fprintf(stderr, "Error: %s", mpg123_strerror(m));
Packit c32a2d
			break; 
Packit c32a2d
		}
Packit c32a2d
	}
Packit c32a2d

Packit c32a2d
	fprintf(stderr, "Finished read %lu, decoded %lu\n", (unsigned long)inc, (unsigned long)outc);
Packit c32a2d

Packit c32a2d
	closewav();
Packit c32a2d
	fclose(out);
Packit c32a2d
	fclose(in);
Packit c32a2d
	mpg123_delete(m);
Packit c32a2d
	mpg123_exit();
Packit c32a2d
	return 0;
Packit c32a2d
}