Fix for autoreconf failures.
[flac.git] / src / flac / utils.c
index ddc64b6..4b7a142 100644 (file)
@@ -1,5 +1,5 @@
 /* flac - Command-line FLAC encoder/decoder
- * Copyright (C) 2002,2003,2004,2005,2006  Josh Coalson
+ * Copyright (C) 2002,2003,2004,2005,2006,2007,2008,2009  Josh Coalson
  *
  * This program is free software; you can redistribute it and/or
  * modify it under the terms of the GNU General Public License
 
 #include "utils.h"
 #include "FLAC/assert.h"
+#include "FLAC/metadata.h"
 #include <math.h>
 #include <stdarg.h>
+#include <stdio.h>
 #include <stdlib.h>
 #include <string.h>
 
+const char *CHANNEL_MASK_TAG = "WAVEFORMATEXTENSIBLE_CHANNEL_MASK";
+
 int flac__utils_verbosity_ = 2;
 
 static FLAC__bool local__parse_uint64_(const char *s, FLAC__uint64 *value)
@@ -51,7 +55,7 @@ static FLAC__bool local__parse_timecode_(const char *s, double *value)
 {
        double ret;
        unsigned i;
-       char c;
+       char c, *endptr;
 
        /* parse [0-9][0-9]*: */
        c = *s++;
@@ -67,15 +71,12 @@ static FLAC__bool local__parse_timecode_(const char *s, double *value)
        }
        ret = (double)i * 60.;
 
-       /* parse [0-9]*[.]?[0-9]* i.e. a sign-less rational number */
-       if(strspn(s, "1234567890.") != strlen(s))
+       /* parse [0-9]*[.,]?[0-9]* i.e. a sign-less rational number (. or , OK for fractional seconds, to support different locales) */
+       if(strspn(s, "1234567890.,") != strlen(s))
+               return false;
+       ret += strtod(s, &endptr);
+       if (endptr == s || *endptr)
                return false;
-       {
-               const char *p = strchr(s, '.');
-               if(p && 0 != strchr(++p, '.'))
-                       return false;
-       }
-       ret += atof(s);
 
        *value = ret;
        return true;
@@ -113,7 +114,7 @@ static FLAC__bool local__parse_cue_(const char *s, const char *end, unsigned *tr
 }
 
 /*
- * @@@ this only works with sorted cuesheets (the spec strongly recommends but
+ * this only works with sorted cuesheets (the spec strongly recommends but
  * does not require sorted cuesheets).  but if it's not sorted, picking a
  * nearest cue point has no significance.
  */
@@ -148,6 +149,11 @@ void flac__utils_printf(FILE *stream, int level, const char *format, ...)
                (void) vfprintf(stream, format, args);
 
                va_end(args);
+
+#ifdef _MSC_VER
+               if(stream == stderr)
+                       fflush(stream); /* for some reason stderr is buffered in at least some if not all MSC libs */
+#endif
        }
 }
 
@@ -269,3 +275,44 @@ void flac__utils_canonicalize_cue_specification(const utils__CueSpecification *c
        else
                until_spec->value.samples = total_samples;
 }
+
+FLAC__bool flac__utils_set_channel_mask_tag(FLAC__StreamMetadata *object, FLAC__uint32 channel_mask)
+{
+       FLAC__StreamMetadata_VorbisComment_Entry entry = { 0, 0 };
+       char tag[128];
+
+       FLAC__ASSERT(object);
+       FLAC__ASSERT(object->type == FLAC__METADATA_TYPE_VORBIS_COMMENT);
+       FLAC__ASSERT(strlen(CHANNEL_MASK_TAG+1+2+16+1) <= sizeof(tag)); /* +1 for =, +2 for 0x, +16 for digits, +1 for NUL */
+       entry.entry = (FLAC__byte*)tag;
+#if defined _MSC_VER || defined __MINGW32__
+       if((entry.length = _snprintf(tag, sizeof(tag), "%s=0x%04X", CHANNEL_MASK_TAG, (unsigned)channel_mask)) >= sizeof(tag))
+#else
+       if((entry.length = snprintf(tag, sizeof(tag), "%s=0x%04X", CHANNEL_MASK_TAG, (unsigned)channel_mask)) >= sizeof(tag))
+#endif
+               return false;
+       if(!FLAC__metadata_object_vorbiscomment_replace_comment(object, entry, /*all=*/true, /*copy=*/true))
+               return false;
+       return true;
+}
+
+FLAC__bool flac__utils_get_channel_mask_tag(const FLAC__StreamMetadata *object, FLAC__uint32 *channel_mask)
+{
+       int offset;
+       unsigned val;
+       char *p;
+       FLAC__ASSERT(object);
+       FLAC__ASSERT(object->type == FLAC__METADATA_TYPE_VORBIS_COMMENT);
+       if(0 > (offset = FLAC__metadata_object_vorbiscomment_find_entry_from(object, /*offset=*/0, CHANNEL_MASK_TAG)))
+               return false;
+       if(object->data.vorbis_comment.comments[offset].length < strlen(CHANNEL_MASK_TAG)+4)
+               return false;
+       if(0 == (p = strchr((const char *)object->data.vorbis_comment.comments[offset].entry, '='))) /* should never happen, but just in case */
+               return false;
+       if(strncmp(p, "=0x", 3))
+               return false;
+       if(sscanf(p+3, "%x", &val) != 1)
+               return false;
+       *channel_mask = val;
+       return true;
+}