summaryrefslogtreecommitdiffstats
path: root/mpg123_artsplugin/dxhead.c
diff options
context:
space:
mode:
Diffstat (limited to 'mpg123_artsplugin/dxhead.c')
-rw-r--r--mpg123_artsplugin/dxhead.c250
1 files changed, 250 insertions, 0 deletions
diff --git a/mpg123_artsplugin/dxhead.c b/mpg123_artsplugin/dxhead.c
new file mode 100644
index 00000000..fb056055
--- /dev/null
+++ b/mpg123_artsplugin/dxhead.c
@@ -0,0 +1,250 @@
+/*---- DXhead.c --------------------------------------------
+
+
+decoder MPEG Layer III
+
+handle Xing header
+
+mod 12/7/98 add vbr scale
+
+Copyright 1998 Xing Technology Corp.
+-----------------------------------------------------------*/
+#include <stdlib.h>
+#include <unistd.h>
+#include <float.h>
+#include <math.h>
+#include "mpg123/mpg123.h"
+#include "dxhead.h"
+
+/* 4 Xing
+ * 4 flags
+ * 4 frames
+ * 4 bytes
+ * 100 toc
+ */
+
+/*-------------------------------------------------------------*/
+static int ExtractI4(unsigned char *buf)
+{
+
+ int x;
+
+/* big endian extract */
+
+ x = buf[0];
+
+ x <<= 8;
+
+ x |= buf[1];
+
+ x <<= 8;
+
+ x |= buf[2];
+
+ x <<= 8;
+
+ x |= buf[3];
+
+
+ return x;
+
+}
+
+/*-------------------------------------------------------------*/
+int mpg123_get_xing_header(XHEADDATA * X, unsigned char *buf)
+{
+
+ int i, head_flags;
+
+ int h_id, h_mode, h_sr_index;
+
+ static int sr_table[4] =
+ {44100, 48000, 32000, 99999};
+
+
+/* get Xing header data */
+
+
+ X->flags = 0; /* clear to null incase fail */
+ X->toc = NULL;
+
+
+/* get selected MPEG header data */
+ h_id = (buf[1] >> 3) & 1;
+
+ h_sr_index = (buf[2] >> 2) & 3;
+
+ h_mode = (buf[3] >> 6) & 3;
+
+
+
+/* determine offset of header */
+ if (h_id)
+ { /* mpeg1 */
+
+ if (h_mode != 3) {
+ buf += (32 + 4);
+
+}
+
+ else
+ buf += (17 + 4);
+
+ }
+
+ else
+ { /* mpeg2 */
+
+ if (h_mode != 3)
+ buf += (17 + 4);
+
+ else
+ buf += (9 + 4);
+
+ }
+
+
+ if (buf[0] != 'X')
+ return 0; /* fail */
+
+ if (buf[1] != 'i')
+ return 0; /* header not found */
+
+ if (buf[2] != 'n')
+ return 0;
+
+ if (buf[3] != 'g')
+ return 0;
+
+ buf += 4;
+
+
+ X->h_id = h_id;
+
+ X->samprate = sr_table[h_sr_index];
+
+ if (h_id == 0)
+ X->samprate >>= 1;
+
+
+ head_flags = X->flags = ExtractI4(buf);
+ buf += 4; /* get flags */
+
+
+ if (head_flags & FRAMES_FLAG)
+ {
+ X->frames = ExtractI4(buf);
+ buf += 4;
+ }
+
+ if (head_flags & BYTES_FLAG)
+ {
+ X->bytes = ExtractI4(buf);
+ buf += 4;
+ }
+
+
+ if (head_flags & TOC_FLAG)
+ {
+
+ X->toc = malloc(100);
+ if (X->toc != NULL)
+ {
+
+ for (i = 0; i < 100; i++)
+ X->toc[i] = buf[i];
+
+ }
+
+ buf += 100;
+
+ }
+
+
+ X->vbr_scale = -1;
+
+ if (head_flags & VBR_SCALE_FLAG)
+ {
+ X->vbr_scale = ExtractI4(buf);
+ buf += 4;
+ }
+
+
+/*if( X->toc != NULL ) {
+ *for(i=0;i<100;i++) {
+ * if( (i%10) == 0 ) printf("\n");
+ * printf(" %3d", (int)(X->toc[i]));
+ *}
+ *}
+ */
+
+ return 1; /* success */
+
+}
+
+/*-------------------------------------------------------------*/
+int mpg123_seek_point(unsigned char TOC[100], int file_bytes, float percent)
+{
+
+/* interpolate in TOC to get file seek point in bytes */
+ int a, seekpoint;
+
+ float fa, fb, fx;
+
+
+
+ if (percent < 0.0f)
+ percent = 0.0f;
+
+ if (percent > 100.0f)
+ percent = 100.0f;
+
+
+ a = (int) percent;
+
+ if (a > 99)
+ a = 99;
+
+ fa = TOC[a];
+
+ if (a < 99)
+ {
+
+ fb = TOC[a + 1];
+
+ }
+
+ else
+ {
+
+ fb = 256.0f;
+
+ }
+
+
+
+ fx = fa + (fb - fa) * (percent - a);
+
+
+ seekpoint = (int) ((1.0f / 256.0f) * fx * file_bytes);
+
+
+
+ return seekpoint;
+
+}
+
+/*-------------------------------------------------------------*/
+int mpg123_stream_check_for_xing_header(struct frame *fr, XHEADDATA * xhead)
+{
+ unsigned char *head_data;
+ int ret;
+
+ lseek(rd->filept, -(fr->framesize + 4), SEEK_CUR);
+ head_data = malloc(fr->framesize + 4);
+ read(rd->filept,head_data, fr->framesize +4); /* now read the rest */
+ ret = mpg123_get_xing_header(xhead, head_data);
+ free(head_data);
+ return ret;
+}
+