forked from len0rd/rockbox
		
	git-svn-id: svn://svn.rockbox.org/rockbox/trunk@12459 a1c6a512-1295-4272-9138-f99709370657
		
			
				
	
	
		
			463 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			463 lines
		
	
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /***************************************************************************
 | |
|  *             __________               __   ___.
 | |
|  *   Open      \______   \ ____   ____ |  | _\_ |__   _______  ___
 | |
|  *   Source     |       _//  _ \_/ ___\|  |/ /| __ \ /  _ \  \/  /
 | |
|  *   Jukebox    |    |   (  <_> )  \___|    < | \_\ (  <_> > <  <
 | |
|  *   Firmware   |____|_  /\____/ \___  >__|_ \|___  /\____/__/\_ \
 | |
|  *                     \/            \/     \/    \/            \/
 | |
|  * $Id$
 | |
|  *
 | |
|  * Copyright (C) 2005 Magnus Holmgren
 | |
|  *
 | |
|  * All files in this archive are subject to the GNU General Public License.
 | |
|  * See the file COPYING in the source tree root for full license agreement.
 | |
|  *
 | |
|  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
 | |
|  * KIND, either express or implied.
 | |
|  *
 | |
|  ****************************************************************************/
 | |
| 
 | |
| #include <ctype.h>
 | |
| #include <inttypes.h>
 | |
| #include <math.h>
 | |
| #include <stdbool.h>
 | |
| #include <stdio.h>
 | |
| #include <stdlib.h>
 | |
| #include <string.h>
 | |
| #include <system.h>
 | |
| #include "id3.h"
 | |
| #include "debug.h"
 | |
| 
 | |
| /* Type of channel for RVA2 frame. There are more than this defined in the spec
 | |
|    but we don't use them. */
 | |
| #define MASTER_CHANNEL 1
 | |
| 
 | |
| /* The fixed point math routines (with the exception of fp_atof) are based
 | |
|  * on oMathFP by Dan Carter (http://orbisstudios.com).
 | |
|  */
 | |
| 
 | |
| /* 12 bits of precision gives fairly accurate result, but still allows a
 | |
|  * compact implementation. The math code supports up to 13...
 | |
|  */
 | |
| 
 | |
| #define FP_BITS         (12)
 | |
| #define FP_MASK         ((1 << FP_BITS) - 1)
 | |
| #define FP_ONE          (1 << FP_BITS)
 | |
| #define FP_TWO          (2 << FP_BITS)
 | |
| #define FP_HALF         (1 << (FP_BITS - 1))
 | |
| #define FP_LN2          ( 45426 >> (16 - FP_BITS))
 | |
| #define FP_LN2_INV      ( 94548 >> (16 - FP_BITS))
 | |
| #define FP_EXP_ZERO     ( 10922 >> (16 - FP_BITS))
 | |
| #define FP_EXP_ONE      (  -182 >> (16 - FP_BITS))
 | |
| #define FP_EXP_TWO      (     4 >> (16 - FP_BITS))
 | |
| #define FP_INF          (0x7fffffff)
 | |
| #define FP_LN10         (150902 >> (16 - FP_BITS))
 | |
| 
 | |
| #define FP_MAX_DIGITS       (4)
 | |
| #define FP_MAX_DIGITS_INT   (10000)
 | |
| 
 | |
| #define FP_FAST_MUL_DIV
 | |
| 
 | |
| #ifdef FP_FAST_MUL_DIV
 | |
| 
 | |
| /* These macros can easily overflow, but they are good enough for our uses,
 | |
|  * and saves some code.
 | |
|  */
 | |
| #define fp_mul(x, y) (((x) * (y)) >> FP_BITS)
 | |
| #define fp_div(x, y) (((x) << FP_BITS) / (y))
 | |
| 
 | |
| #else
 | |
| 
 | |
| static long fp_mul(long x, long y)
 | |
| {
 | |
|     long x_neg = 0;
 | |
|     long y_neg = 0;
 | |
|     long rc;
 | |
| 
 | |
|     if ((x == 0) || (y == 0))
 | |
|     {
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     if (x < 0)
 | |
|     {
 | |
|         x_neg = 1;
 | |
|         x = -x;
 | |
|     }
 | |
| 
 | |
|     if (y < 0)
 | |
|     {
 | |
|         y_neg = 1;
 | |
|         y = -y;
 | |
|     }
 | |
| 
 | |
|     rc = (((x >> FP_BITS) * (y >> FP_BITS)) << FP_BITS)
 | |
|         + (((x & FP_MASK) * (y & FP_MASK)) >> FP_BITS)
 | |
|         + ((x & FP_MASK) * (y >> FP_BITS))
 | |
|         + ((x >> FP_BITS) * (y & FP_MASK));
 | |
| 
 | |
|     if ((x_neg ^ y_neg) == 1)
 | |
|     {
 | |
|         rc = -rc;
 | |
|     }
 | |
| 
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| static long fp_div(long x, long y)
 | |
| {
 | |
|     long x_neg = 0;
 | |
|     long y_neg = 0;
 | |
|     long shifty;
 | |
|     long rc;
 | |
|     int msb = 0;
 | |
|     int lsb = 0;
 | |
| 
 | |
|     if (x == 0)
 | |
|     {
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     if (y == 0)
 | |
|     {
 | |
|         return (x < 0) ? -FP_INF : FP_INF;
 | |
|     }
 | |
| 
 | |
|     if (x < 0)
 | |
|     {
 | |
|         x_neg = 1;
 | |
|         x = -x;
 | |
|     }
 | |
| 
 | |
|     if (y < 0)
 | |
|     {
 | |
|         y_neg = 1;
 | |
|         y = -y;
 | |
|     }
 | |
| 
 | |
|     while ((x & (1 << (30 - msb))) == 0)
 | |
|     {
 | |
|         msb++;
 | |
|     }
 | |
| 
 | |
|     while ((y & (1 << lsb)) == 0)
 | |
|     {
 | |
|         lsb++;
 | |
|     }
 | |
| 
 | |
|     shifty = FP_BITS - (msb + lsb);
 | |
|     rc = ((x << msb) / (y >> lsb));
 | |
| 
 | |
|     if (shifty > 0)
 | |
|     {
 | |
|         rc <<= shifty;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         rc >>= -shifty;
 | |
|     }
 | |
| 
 | |
|     if ((x_neg ^ y_neg) == 1)
 | |
|     {
 | |
|         rc = -rc;
 | |
|     }
 | |
| 
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| #endif /* FP_FAST_MUL_DIV */
 | |
| 
 | |
| static long fp_exp(long x)
 | |
| {
 | |
|     long k;
 | |
|     long z;
 | |
|     long R;
 | |
|     long xp;
 | |
| 
 | |
|     if (x == 0)
 | |
|     {
 | |
|         return FP_ONE;
 | |
|     }
 | |
| 
 | |
|     k = (fp_mul(abs(x), FP_LN2_INV) + FP_HALF) & ~FP_MASK;
 | |
| 
 | |
|     if (x < 0)
 | |
|     {
 | |
|         k = -k;
 | |
|     }
 | |
| 
 | |
|     x -= fp_mul(k, FP_LN2);
 | |
|     z = fp_mul(x, x);
 | |
|     R = FP_TWO + fp_mul(z, FP_EXP_ZERO + fp_mul(z, FP_EXP_ONE
 | |
|         + fp_mul(z, FP_EXP_TWO)));
 | |
|     xp = FP_ONE + fp_div(fp_mul(FP_TWO, x), R - x);
 | |
| 
 | |
|     if (k < 0)
 | |
|     {
 | |
|         k = FP_ONE >> (-k >> FP_BITS);
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         k = FP_ONE << (k >> FP_BITS);
 | |
|     }
 | |
| 
 | |
|     return fp_mul(k, xp);
 | |
| }
 | |
| 
 | |
| static long fp_exp10(long x)
 | |
| {
 | |
|     if (x == 0)
 | |
|     {
 | |
|         return FP_ONE;
 | |
|     }
 | |
| 
 | |
|     return fp_exp(fp_mul(FP_LN10, x));
 | |
| }
 | |
| 
 | |
| static long fp_atof(const char* s, int precision)
 | |
| {
 | |
|     long int_part = 0;
 | |
|     long int_one = 1 << precision;
 | |
|     long frac_part = 0;
 | |
|     long frac_count = 0;
 | |
|     long frac_max = ((precision * 4) + 12) / 13;
 | |
|     long frac_max_int = 1;
 | |
|     long sign = 1;
 | |
|     bool point = false;
 | |
| 
 | |
|     while ((*s != '\0') && isspace(*s))
 | |
|     {
 | |
|         s++;
 | |
|     }
 | |
| 
 | |
|     if (*s == '-')
 | |
|     {
 | |
|         sign = -1;
 | |
|         s++;
 | |
|     }
 | |
|     else if (*s == '+')
 | |
|     {
 | |
|         s++;
 | |
|     }
 | |
| 
 | |
|     while (*s != '\0')
 | |
|     {
 | |
|         if (*s == '.')
 | |
|         {
 | |
|             if (point)
 | |
|             {
 | |
|                 break;
 | |
|             }
 | |
| 
 | |
|             point = true;
 | |
|         }
 | |
|         else if (isdigit(*s))
 | |
|         {
 | |
|             if (point)
 | |
|             {
 | |
|                 if (frac_count < frac_max)
 | |
|                 {
 | |
|                     frac_part = frac_part * 10 + (*s - '0');
 | |
|                     frac_count++;
 | |
|                     frac_max_int *= 10;
 | |
|                 }
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 int_part = int_part * 10 + (*s - '0');
 | |
|             }
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             break;
 | |
|         }
 | |
| 
 | |
|         s++;
 | |
|     }
 | |
| 
 | |
|     while (frac_count < frac_max)
 | |
|     {
 | |
|       frac_part *= 10;
 | |
|       frac_count++;
 | |
|       frac_max_int *= 10;
 | |
|     }
 | |
| 
 | |
|     return sign * ((int_part * int_one)
 | |
|         + (((int64_t) frac_part * int_one) / frac_max_int));
 | |
| }
 | |
| 
 | |
| static long convert_gain(long gain)
 | |
| {
 | |
|     /* Don't allow unreasonably low or high gain changes.
 | |
|      * Our math code can't handle it properly anyway. :)
 | |
|      */
 | |
|     if (gain < (-48 * FP_ONE))
 | |
|     {
 | |
|         gain = -48 * FP_ONE;
 | |
|     }
 | |
| 
 | |
|     if (gain > (17 * FP_ONE))
 | |
|     {
 | |
|         gain = 17 * FP_ONE;
 | |
|     }
 | |
| 
 | |
|     gain = fp_exp10(gain / 20) << (24 - FP_BITS);
 | |
| 
 | |
|     return gain;
 | |
| }
 | |
| 
 | |
| long get_replaygain_int(long int_gain)
 | |
| {
 | |
|     return convert_gain(int_gain * FP_ONE / 100);
 | |
| }
 | |
| 
 | |
| long get_replaygain(const char* str)
 | |
| {
 | |
|     long gain = 0;
 | |
| 
 | |
|     if (str)
 | |
|     {
 | |
|         gain = fp_atof(str, FP_BITS);
 | |
|         gain = convert_gain(gain);
 | |
|     }
 | |
| 
 | |
|     return gain;
 | |
| }
 | |
| 
 | |
| long get_replaypeak(const char* str)
 | |
| {
 | |
|     long peak = 0;
 | |
| 
 | |
|     if (str)
 | |
|     {
 | |
|         peak = fp_atof(str, 24);
 | |
|     }
 | |
| 
 | |
|     return peak;
 | |
| }
 | |
| 
 | |
| /* Check for a ReplayGain tag conforming to the "VorbisGain standard". If
 | |
|  * found, set the mp3entry accordingly. buffer is where to store the text
 | |
|  * contents of the gain tags; up to length bytes (including end nil) can be
 | |
|  * written. Returns number of bytes written to the tag text buffer, or zero
 | |
|  * if no ReplayGain tag was found (or nothing was copied to the buffer for
 | |
|  * other reasons).
 | |
|  */
 | |
| long parse_replaygain(const char* key, const char* value,
 | |
|     struct mp3entry* entry, char* buffer, int length)
 | |
| {
 | |
|     char **p = NULL;
 | |
| 
 | |
|     if (((strcasecmp(key, "replaygain_track_gain") == 0)
 | |
|         || (strcasecmp(key, "rg_radio") == 0)) && !entry->track_gain)
 | |
|     {
 | |
|         entry->track_gain = get_replaygain(value);
 | |
|         p = &(entry->track_gain_string);
 | |
|     }
 | |
|     else if (((strcasecmp(key, "replaygain_album_gain") == 0)
 | |
|         || (strcasecmp(key, "rg_audiophile") == 0)) && !entry->album_gain)
 | |
|     {
 | |
|         entry->album_gain = get_replaygain(value);
 | |
|         p = &(entry->album_gain_string);
 | |
|     }
 | |
|     else if (((strcasecmp(key, "replaygain_track_peak") == 0)
 | |
|         || (strcasecmp(key, "rg_peak") == 0)) && !entry->track_peak)
 | |
|     {
 | |
|         entry->track_peak = get_replaypeak(value);
 | |
|     }
 | |
|     else if ((strcasecmp(key, "replaygain_album_peak") == 0)
 | |
|         && !entry->album_peak)
 | |
|     {
 | |
|         entry->album_peak = get_replaypeak(value);
 | |
|     }
 | |
| 
 | |
|     if (p)
 | |
|     {
 | |
|         int len = strlen(value);
 | |
| 
 | |
|         len = MIN(len, length - 1);
 | |
| 
 | |
|         /* A few characters just isn't interesting... */
 | |
|         if (len > 1)
 | |
|         {
 | |
|             strncpy(buffer, value, len);
 | |
|             buffer[len] = 0;
 | |
|             *p = buffer;
 | |
|             return len + 1;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| static long get_rva_values(const char *frame, long *gain, long *peak,
 | |
|     char **string, char *buffer, int length)
 | |
| {
 | |
|     long value, len;
 | |
|     int negative = 0;
 | |
|     char tmpbuf[10];
 | |
|     int peakbits, peakbytes, shift;
 | |
|     unsigned long peakvalue = 0;
 | |
| 
 | |
|     value = 256 * ((unsigned char)*frame) + ((unsigned char)*(frame + 1));
 | |
|     if (value & 0x8000)
 | |
|     {
 | |
|         value = -(value | ~0xFFFF);
 | |
|         negative = 1;
 | |
|     }
 | |
| 
 | |
|     len = snprintf(tmpbuf, sizeof(tmpbuf), "%s%d.%02d dB", negative ? "-" : "",
 | |
|         value / 512, (value & 0x1FF) * 195 / 1000);
 | |
| 
 | |
|     *gain = get_replaygain(tmpbuf);
 | |
| 
 | |
|     len = MIN(len, length - 1);
 | |
|     if (len > 1)
 | |
|     {
 | |
|         strncpy(buffer, tmpbuf, len);
 | |
|         buffer[len] = 0;
 | |
|         *string = buffer;
 | |
|     }
 | |
| 
 | |
|     frame += 2;
 | |
|     peakbits = *(unsigned char *)frame++;
 | |
|     peakbytes = MIN(4, (peakbits + 7) >> 3);
 | |
|     shift = ((8 - (peakbits & 7)) & 7) + (4 - peakbytes) * 8;
 | |
| 
 | |
|     for (; peakbytes; peakbytes--)
 | |
|     {
 | |
|             peakvalue <<= 8;
 | |
|             peakvalue += (unsigned long)*frame++;
 | |
|     }
 | |
| 
 | |
|     peakvalue <<= shift;
 | |
| 
 | |
|     if (peakbits > 32)
 | |
|         peakvalue += (unsigned long)*frame >> (8 - shift);
 | |
| 
 | |
|     snprintf(tmpbuf, sizeof(tmpbuf), "%d.%06d", peakvalue >> 31,
 | |
|         (peakvalue & ~(1 << 31)) / 2147);
 | |
| 
 | |
|     *peak = get_replaypeak(tmpbuf);
 | |
| 
 | |
|     return len + 1;
 | |
| }
 | |
| 
 | |
| long parse_replaygain_rva(const char* key, const char* value,
 | |
|     struct mp3entry* entry, char* buffer, int length)
 | |
| {
 | |
|     /* Values will be overwritten if they already exist. This gives priority to
 | |
|        replaygain in RVA2 fields over TXXX fields for ID3v2.4. */
 | |
|     if ((strcasecmp(key, "track") == 0) && *value == MASTER_CHANNEL)
 | |
|     {
 | |
|         return get_rva_values(value + 1, &(entry->track_gain), &(entry->track_peak),
 | |
|             &(entry->track_gain_string), buffer, length);
 | |
|     }
 | |
|     else if ((strcasecmp(key, "album") == 0) && *value == MASTER_CHANNEL)
 | |
|     {
 | |
|         return get_rva_values(value + 1, &(entry->album_gain), &(entry->album_peak),
 | |
|             &(entry->album_gain_string), buffer, length);
 | |
|     }
 | |
| 
 | |
|     return 0;
 | |
| }
 |