FreeCalypso > hg > freecalypso-tools
view ringtools/fc-e1decode.c @ 308:6a254cc6a7f3
doc/Host-tools-overview: fc-cal2bin documented
author | Mychaela Falconia <falcon@freecalypso.org> |
---|---|
date | Wed, 22 Nov 2017 04:47:36 +0000 |
parents | c8806a5d4a6a |
children |
line wrap: on
line source
/* * This program decodes a binary Melody E1 file into the ASCII source format * of our own invention which fc-e1gen accepts as input. */ #include <sys/types.h> #include <stdio.h> #include <stdlib.h> #include <ctype.h> #include <string.h> #include <strings.h> char *infname; FILE *inf, *outf; static unsigned get_word() { u_char b[2]; int i, c; for (i = 0; i < 2; i++) { c = getc(inf); if (c < 0) { fprintf(stderr, "error: premature EOF in %s\n", infname); exit(1); } b[i] = c; } return((b[1] << 8) | b[0]); } do_global_osc_set() { unsigned word, osc; word = get_word(); fprintf(outf, "osc-set (%u)", word & 0xFF); for (osc = 0; osc < 8; osc++) if (word & (1 << (osc + 8))) fprintf(outf, " %u", osc); putc('\n', outf); putc('\n', outf); } convert_signed_14bit(u) unsigned u; { int i; i = u; if (i >= 8192) i -= 16384; return (i); } convert_signed_5bit(u) unsigned u; { int i; i = u; if (i >= 16) i -= 32; return (i); } process_oscwords(osc, warnflags) unsigned osc, *warnflags; { unsigned word0, word1, extraword; word0 = get_word(); word1 = get_word(); if (word1 & 0xF) *warnflags |= 1; fprintf(outf, "osc %u (%u)", osc, word0 & 1); if (word0 & 2) { fprintf(outf, " df %d %u", convert_signed_14bit(word0 >> 2), word1 >> 6); } else { if (word0 & 4) fputs(" sq1", outf); if (word0 & 8) fputs(" sq2", outf); fprintf(outf, " %u %u %u", (word0 >> 4) & 0x3F, word0 >> 10, word1 >> 6); } if (word1 & 0x10) { extraword = get_word(); if (extraword & 0xFF) *warnflags |= 2; fprintf(outf, " trem %u %d", (extraword >> 8) & 7, convert_signed_5bit(extraword >> 11)); } if (word1 & 0x20) { extraword = get_word(); fprintf(outf, " env %u %u %u %u %u", extraword & 0xF, extraword >> 13, (extraword >> 10) & 7, (extraword >> 7) & 7, (extraword >> 4) & 7); } putc('\n', outf); } process_block(timeword) unsigned timeword; { unsigned osc, warnflags[8], anywarn; fprintf(outf, "time %u\n", timeword & 0xFF); for (osc = 0; osc < 8; osc++) { warnflags[osc] = 0; if (timeword & (1 << (osc + 8))) process_oscwords(osc, warnflags + osc); } putc('\n', outf); anywarn = 0; for (osc = 0; osc < 8; osc++) { if (warnflags[osc] & 1) { fprintf(outf, "# warning: reserved bits set in osc %u word1 above\n", osc); anywarn = 1; } if (warnflags[osc] & 2) { fprintf(outf, "# warning: reserved bits set in osc %u word2 above\n", osc); anywarn = 1; } } if (anywarn) putc('\n', outf); } main(argc, argv) char **argv; { unsigned timeword; if (argc < 2 || argc > 3) { fprintf(stderr, "usage: %s infile [outfile]\n", argv[0]); exit(1); } infname = argv[1]; inf = fopen(infname, "r"); if (!inf) { perror(infname); exit(1); } if (argc > 2) { outf = fopen(argv[2], "w"); if (!outf) { perror(argv[2]); exit(1); } } else outf = stdout; do_global_osc_set(); for (;;) { timeword = get_word(); if (timeword == 0) break; process_block(timeword); } fputs("end\n", outf); exit(0); }