Mercurial > audlegacy-plugins
view src/mpg123/dxhead.c @ 189:772ad5bc7050 trunk
[svn] - now working on amd64/little-endian. no idea how it was working in bigendian mode, though.
author | nenolod |
---|---|
date | Thu, 02 Nov 2006 18:25:48 -0800 |
parents | 3da1b8942b8b |
children |
line wrap: on
line source
/* * Handle Xing vbr header */ #include "config.h" #include "dxhead.h" #include <stdlib.h> #include <float.h> #include <math.h> #include <string.h> #include <glib.h> #define GET_INT32BE(b) \ (i = (b[0] << 24) | (b[1] << 16) | b[2] << 8 | b[3], b += 4, i) int mpgdec_get_xing_header(xing_header_t * xing, unsigned char *buf) { int i, head_flags; int id, mode; memset(xing, 0, sizeof(xing_header_t)); /* get selected MPEG header data */ id = (buf[1] >> 3) & 1; mode = (buf[3] >> 6) & 3; buf += 4; /* Skip the sub band data */ if (id) { /* mpeg1 */ if (mode != 3) buf += 32; else buf += 17; } else { /* mpeg2 */ if (mode != 3) buf += 17; else buf += 9; } if (strncmp((char *) buf, "Xing", 4)) return 0; buf += 4; head_flags = GET_INT32BE(buf); if (head_flags & FRAMES_FLAG) xing->frames = GET_INT32BE(buf); if (xing->frames < 1) return 0; if (head_flags & BYTES_FLAG) xing->bytes = GET_INT32BE(buf); if (head_flags & TOC_FLAG) { for (i = 0; i < 100; i++) { xing->toc[i] = buf[i]; if (i > 0 && xing->toc[i] < xing->toc[i - 1]) return 0; } if (xing->toc[99] == 0) return 0; buf += 100; } else for (i = 0; i < 100; i++) xing->toc[i] = (i * 256) / 100; #ifdef XING_DEBUG for (i = 0; i < 100; i++) { if ((i % 10) == 0) fprintf(stderr, "\n"); fprintf(stderr, " %3d", xing->toc[i]); } #endif return 1; } int mpgdec_seek_point(xing_header_t * xing, float percent) { /* interpolate in TOC to get file seek point in bytes */ int a, seekpoint; float fa, fb, fx; percent = CLAMP(percent, 0, 100); a = MIN(percent, 99); fa = xing->toc[a]; if (a < 99) fb = xing->toc[a + 1]; else fb = 256; fx = fa + (fb - fa) * (percent - a); seekpoint = (1.0f / 256.0f) * fx * xing->bytes; return seekpoint; }