forked from len0rd/rockbox
		
	be rewritten to fixed point. seems to compile cleanly for iriver. git-svn-id: svn://svn.rockbox.org/rockbox/trunk@6197 a1c6a512-1295-4272-9138-f99709370657
		
			
				
	
	
		
			335 lines
		
	
	
	
		
			8.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			335 lines
		
	
	
	
		
			8.3 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*  _______         ____    __         ___    ___
 | |
|  * \    _  \       \    /  \  /       \   \  /   /       '   '  '
 | |
|  *  |  | \  \       |  |    ||         |   \/   |         .      .
 | |
|  *  |  |  |  |      |  |    ||         ||\  /|  |
 | |
|  *  |  |  |  |      |  |    ||         || \/ |  |         '  '  '
 | |
|  *  |  |  |  |      |  |    ||         ||    |  |         .      .
 | |
|  *  |  |_/  /        \  \__//          ||    |  |
 | |
|  * /_______/ynamic    \____/niversal  /__\  /____\usic   /|  .  . ibliotheque
 | |
|  *                                                      /  \
 | |
|  *                                                     / .  \
 | |
|  * dumbout.c - Utility to stream music to a file.     / / \  \
 | |
|  *                                                   | <  /   \_
 | |
|  * By entheh.                                        |  \/ /\   /
 | |
|  *                                                    \_  /  > /
 | |
|  *                                                      | \ / /
 | |
|  *                                                      |  ' /
 | |
|  *                                                       \__/
 | |
|  */
 | |
| 
 | |
| #include <time.h>
 | |
| #include <stdlib.h>
 | |
| #include <stdio.h>
 | |
| #include <ctype.h>
 | |
| #include <math.h>
 | |
| #include <string.h>
 | |
| #include <dumb.h>
 | |
| 
 | |
| 
 | |
| union {
 | |
| 	short s16[8192];
 | |
| 	char s8[16384];
 | |
| } buffer;
 | |
| 
 | |
| 
 | |
| int main(int argc, const char *const *argv) /* I'm const-crazy! */
 | |
| {
 | |
| 	DUH *duh;
 | |
| 	DUH_SIGRENDERER *sr;
 | |
| 
 | |
| 	const char *fn = NULL;
 | |
| 	const char *fn_out = NULL;
 | |
| 	FILE *outf;
 | |
| 
 | |
| 	int depth = 16;
 | |
| 	int bigendian = 0;
 | |
| 	int unsign = 0;
 | |
| 	int freq = 44100;
 | |
| 	int n_channels = 2;
 | |
| 	float volume = 1.0f;
 | |
| 	float delay = 0.0f;
 | |
| 	float delta;
 | |
| 	int bufsize;
 | |
| 	clock_t start, end;
 | |
| 
 | |
| 	int i = 1;
 | |
| 
 | |
| 	LONG_LONG length;
 | |
| 	LONG_LONG done;
 | |
| 	int dots;
 | |
| 
 | |
| 	while (i < argc) {
 | |
| 		const char *arg = argv[i++];
 | |
| 		if (*arg != '-') {
 | |
| 			if (fn) {
 | |
| 				fprintf(stderr,
 | |
| 					"Cannot specify multiple filenames!\n"
 | |
| 					"Second filename found: \"%s\"\n", arg);
 | |
| 				return 1;
 | |
| 			}
 | |
| 			fn = arg;
 | |
| 			continue;
 | |
| 		}
 | |
| 		arg++;
 | |
| 		while (*arg) {
 | |
| 			char *endptr;
 | |
| 			switch (*arg++) {
 | |
| 				case 'o':
 | |
| 				case 'O':
 | |
| 					if (i >= argc) {
 | |
| 						fprintf(stderr, "Out of arguments; output filename expected!\n");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					fn_out = argv[i++];
 | |
| 					break;
 | |
| 				case 'd':
 | |
| 				case 'D':
 | |
| 					if (i >= argc) {
 | |
| 						fprintf(stderr, "Out of arguments; delay expected!\n");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					delay = (float)strtod(argv[i++], &endptr);
 | |
| 					if (*endptr != 0 || delay < 0.0f || delay > 64.0f) {
 | |
| 						fprintf(stderr, "Invalid delay!\n");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					break;
 | |
| 				case 'v':
 | |
| 				case 'V':
 | |
| 					if (i >= argc) {
 | |
| 						fprintf(stderr, "Out of arguments; volume expected!\n");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					volume = (float)strtod(argv[i++], &endptr);
 | |
| 					if (*endptr != 0 || volume < -8.0f || volume > 8.0f) {
 | |
| 						fprintf(stderr, "Invalid volume!\n");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					break;
 | |
| 				case 's':
 | |
| 				case 'S':
 | |
| 					if (i >= argc) {
 | |
| 						fprintf(stderr, "Out of arguments; sampling rate expected!\n");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					freq = strtol(argv[i++], &endptr, 10);
 | |
| 					if (*endptr != 0 || freq < 1 || freq > 960000) {
 | |
| 						fprintf(stderr, "Invalid sampling rate!\n");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					break;
 | |
| 				case '8':
 | |
| 					depth = 8;
 | |
| 					break;
 | |
| 				case 'b':
 | |
| 				case 'B':
 | |
| 					bigendian = 1;
 | |
| 					break;
 | |
| 				case 'm':
 | |
| 				case 'M':
 | |
| 					n_channels = 1;
 | |
| 					break;
 | |
| 				case 'u':
 | |
| 				case 'U':
 | |
| 					unsign = 1;
 | |
| 					break;
 | |
| 				case 'r':
 | |
| 				case 'R':
 | |
| 					if (i >= argc) {
 | |
| 						fprintf(stderr, "Out of arguments; resampling quality expected!\n");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					dumb_resampling_quality = strtol(argv[i++], &endptr, 10);
 | |
| 					if (*endptr != 0 || dumb_resampling_quality < 0 || dumb_resampling_quality > 2) {
 | |
| 						fprintf(stderr, "Invalid resampling quality!\n");
 | |
| 						return 1;
 | |
| 					}
 | |
| 					break;
 | |
| 				default:
 | |
| 					fprintf(stderr, "Invalid switch - '%c'!\n", isprint(arg[-1]) ? arg[-1] : '?');
 | |
| 					return 1;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (!fn) {
 | |
| 		fprintf(stderr,
 | |
| 			"Usage: dumbout [options] module [more-options]\n"
 | |
| 			"\n"
 | |
| 			"The module can be any IT, XM, S3M or MOD file. It will be rendered to a .pcm\n"
 | |
| 			"file of the same name, unless you specify otherwise with the -o option.\n"
 | |
| 			"\n"
 | |
| 			"The valid options are:\n"
 | |
| 			"-o <file>   specify the output filename (defaults to the input filename with\n"
 | |
| 			"              the extension replaced with .pcm); use - to write to standard\n"
 | |
| 			"              output or . to write nowhere (useful for measuring DUMB's\n"
 | |
| 			"              performance, and DOS and Windows don't have /dev/null!)\n"
 | |
| 			"-d <delay>  set the initial delay, in seconds (default 0.0)\n"
 | |
| 			"-v <volume> adjust the volume (default 1.0)\n"
 | |
| 			"-s <freq>   set the sampling rate in Hz (default 44100)\n"
 | |
| 			"-8          generate 8-bit instead of 16-bit\n"
 | |
| 			"-b          generate big-endian data instead of little-endian (meaningless when\n"
 | |
| 			"              using -8)\n"
 | |
| 			"-m          generate mono output instead of stereo left/right pairs\n"
 | |
| 			"-u          generated unsigned output instead of signed\n"
 | |
| 			"-r <value>  specify the resampling quality to use\n");
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	atexit(&dumb_exit);
 | |
| 	dumb_register_stdfiles();
 | |
| 
 | |
| 	dumb_it_max_to_mix = 256;
 | |
| 
 | |
| 	duh = load_duh(fn);
 | |
| 	if (!duh) {
 | |
| 		duh = dumb_load_it(fn);
 | |
| 		if (!duh) {
 | |
| 			duh = dumb_load_xm(fn);
 | |
| 			if (!duh) {
 | |
| 				duh = dumb_load_s3m(fn);
 | |
| 				if (!duh) {
 | |
| 					duh = dumb_load_mod(fn);
 | |
| 					if (!duh) {
 | |
| 						fprintf(stderr, "Unable to open %s!\n", fn);
 | |
| 						return 1;
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	sr = duh_start_sigrenderer(duh, 0, n_channels, 0);
 | |
| 	if (!sr) {
 | |
| 		unload_duh(duh);
 | |
| 		fprintf(stderr, "Unable to play file!\n");
 | |
| 		return 1;
 | |
| 	}
 | |
| 
 | |
| 	if (fn_out) {
 | |
| 		if (fn_out[0] == '-' && fn_out[1] == 0)
 | |
| 			outf = stdout;
 | |
| 		else if (fn_out[0] == '.' && fn_out[1] == 0)
 | |
| 			outf = NULL;
 | |
| 		else {
 | |
| 			outf = fopen(fn_out, "wb");
 | |
| 			if (!outf) {
 | |
| 				fprintf(stderr, "Unable to open %s for writing!\n", fn_out);
 | |
| 				duh_end_sigrenderer(sr);
 | |
| 				unload_duh(duh);
 | |
| 				return 1;
 | |
| 			}
 | |
| 		}
 | |
| 	} else {
 | |
| 		char *extptr = NULL, *p;
 | |
| 		char *fn_out = malloc(strlen(fn)+5);
 | |
| 		if (!fn_out) {
 | |
| 			fprintf(stderr, "Out of memory!\n");
 | |
| 			duh_end_sigrenderer(sr);
 | |
| 			unload_duh(duh);
 | |
| 			return 1;
 | |
| 		}
 | |
| 		strcpy(fn_out, fn);
 | |
| 		for (p = fn_out; *p; p++)
 | |
| 			if (*p == '.') extptr = p;
 | |
| 		if (!extptr) extptr = p;
 | |
| 		strcpy(extptr, ".pcm");
 | |
| 		outf = fopen(fn_out, "wb");
 | |
| 		if (!outf) {
 | |
| 			fprintf(stderr, "Unable to open %s for writing!\n", fn_out);
 | |
| 			free(fn_out);
 | |
| 			duh_end_sigrenderer(sr);
 | |
| 			unload_duh(duh);
 | |
| 			return 1;
 | |
| 		}
 | |
| 		free(fn_out);
 | |
| 	}
 | |
| 
 | |
| 	{
 | |
| 		DUMB_IT_SIGRENDERER *itsr = duh_get_it_sigrenderer(sr);
 | |
| 		dumb_it_set_loop_callback(itsr, &dumb_it_callback_terminate, NULL);
 | |
| 		dumb_it_set_xm_speed_zero_callback(itsr, &dumb_it_callback_terminate, NULL);
 | |
| 	}
 | |
| 
 | |
| 	length = (LONG_LONG)duh_get_length(duh) * freq >> 16;
 | |
| 	done = 0;
 | |
| 	dots = 0;
 | |
| 	delta = 65536.0f / freq;
 | |
| 	bufsize = depth == 16 ? 8192 : 16384;
 | |
| 	bufsize /= n_channels;
 | |
| 
 | |
| 	{
 | |
| 		long l = (long)floor(delay * freq + 0.5f);
 | |
| 		l *= n_channels * (depth >> 3);
 | |
| 		if (l) {
 | |
| 			if (unsign) {
 | |
| 				if (depth == 16) {
 | |
| 					if (bigendian) {
 | |
| 						for (i = 0; i < 8192; i++) {
 | |
| 							buffer.s8[i*2] = 0x80;
 | |
| 							buffer.s8[i*2+1] = 0x00;
 | |
| 						}
 | |
| 					} else {
 | |
| 						for (i = 0; i < 8192; i++) {
 | |
| 							buffer.s8[i*2] = 0x00;
 | |
| 							buffer.s8[i*2+1] = 0x80;
 | |
| 						}
 | |
| 					}
 | |
| 				} else
 | |
| 					memset(buffer.s8, 0x80, 16384);
 | |
| 			} else
 | |
| 				memset(buffer.s8, 0, 16384);
 | |
| 			while (l >= 16384) {
 | |
| 				if (outf) fwrite(buffer.s8, 1, 16384, outf);
 | |
| 				l -= 16384;
 | |
| 			}
 | |
| 			if (l && outf) fwrite(buffer.s8, 1, l, outf);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	start = clock();
 | |
| 
 | |
| 	fprintf(stderr, "................................................................\n");
 | |
| 	for (;;) {
 | |
| 		int l = duh_render(sr, depth, unsign, volume, delta, bufsize, &buffer);
 | |
| 		if (depth == 16) {
 | |
| 			if (bigendian) {
 | |
| 				for (i = 0; i < l * n_channels; i++) {
 | |
| 					short val = buffer.s16[i];
 | |
| 					buffer.s8[i*2] = val >> 8;
 | |
| 					buffer.s8[i*2+1] = val;
 | |
| 				}
 | |
| 			} else {
 | |
| 				for (i = 0; i < l * n_channels; i++) {
 | |
| 					short val = buffer.s16[i];
 | |
| 					buffer.s8[i*2] = val;
 | |
| 					buffer.s8[i*2+1] = val >> 8;
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		if (outf) fwrite(buffer.s8, 1, l * n_channels * (depth >> 3), outf);
 | |
| 		if (l < bufsize) break;
 | |
| 		done += l;
 | |
| 		l = done * 64 / length;
 | |
| 		while (dots < 64 && l > dots) {
 | |
| 			fprintf(stderr, "|");
 | |
| 			dots++;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	while (64 > dots) {
 | |
| 		fprintf(stderr, "|");
 | |
| 		dots++;
 | |
| 	}
 | |
| 	fprintf(stderr, "\n");
 | |
| 
 | |
| 	end = clock();
 | |
| 
 | |
| 	duh_end_sigrenderer(sr);
 | |
| 	unload_duh(duh);
 | |
| 	if (outf && outf != stdout) fclose(outf);
 | |
| 
 | |
| 	fprintf(stderr, "Elapsed time: %f seconds\n", (end - start) / (float)CLOCKS_PER_SEC);
 | |
| 
 | |
| 	return 0;
 | |
| }
 |