All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.drew.metadata.mp3.Mp3Reader Maven / Gradle / Ivy

/*
 * Copyright 2002-2019 Drew Noakes and contributors
 *
 *    Licensed under the Apache License, Version 2.0 (the "License");
 *    you may not use this file except in compliance with the License.
 *    You may obtain a copy of the License at
 *
 *        http://www.apache.org/licenses/LICENSE-2.0
 *
 *    Unless required by applicable law or agreed to in writing, software
 *    distributed under the License is distributed on an "AS IS" BASIS,
 *    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 *    See the License for the specific language governing permissions and
 *    limitations under the License.
 *
 * More information about this project is available at:
 *
 *    https://drewnoakes.com/code/exif/
 *    https://github.com/drewnoakes/metadata-extractor
 */
package com.drew.metadata.mp3;

import com.drew.imaging.ImageProcessingException;
import com.drew.lang.SequentialReader;
import com.drew.lang.StreamReader;
import com.drew.lang.annotations.NotNull;
import com.drew.metadata.Metadata;

import java.io.IOException;
import java.io.InputStream;

/**
 * Sources: http://id3.org/mp3Frame
 *          https://www.loc.gov/preservation/digital/formats/fdd/fdd000105.shtml
 *
 * @author Payton Garland
 */
public class Mp3Reader
{
    public void extract(@NotNull final InputStream inputStream, @NotNull final Metadata metadata)
    {
        Mp3Directory directory = new Mp3Directory();
        metadata.addDirectory(directory);

        try {
            SequentialReader reader = new StreamReader(inputStream);

            int header = reader.getInt32();

            // ID: MPEG-2.5, MPEG-2, or MPEG-1
            int id = 0;
            switch ((header & 0x000180000) >> 19) {
                case 0:
                    throw new ImageProcessingException("MPEG-2.5 not supported.");
                case 2:
                    directory.setString(Mp3Directory.TAG_ID, "MPEG-2");
                    id = 2;
                    break;
                case 3:
                    directory.setString(Mp3Directory.TAG_ID, "MPEG-1");
                    id = 1;
                    break;
            }

            // Layer Type: 1, 2, 3, or not defined
            int layer = ((header & 0x00060000) >> 17);
            switch (layer) {
                case 0:
                    directory.setString(Mp3Directory.TAG_LAYER, "Not defined");
                    break;
                case 1:
                    directory.setString(Mp3Directory.TAG_LAYER, "Layer III");
                    break;
                case 2:
                    directory.setString(Mp3Directory.TAG_LAYER, "Layer II");
                    break;
                case 3:
                    directory.setString(Mp3Directory.TAG_LAYER, "Layer I");
                    break;
            }


            int protectionBit = ((header & 0x00010000) >> 16);

            // Bitrate: depends on ID and Layer
            int bitrate = ((header & 0x0000F000) >> 12);
            if (bitrate != 0 && bitrate != 15)
                directory.setInt(Mp3Directory.TAG_BITRATE, setBitrate(bitrate, layer, id));

            // Frequency: depends on ID
            int frequency = ((header & 0x00000C00) >> 10);
            int[][] frequencyMapping = new int[2][3];
            frequencyMapping[0] = new int[]{44100, 48000, 32000};
            frequencyMapping[1] = new int[]{22050, 24000, 16000};
            if (id == 2) {
                directory.setInt(Mp3Directory.TAG_FREQUENCY, frequencyMapping[1][frequency]);
                frequency = frequencyMapping[1][frequency];
            } else if (id == 1) {
                directory.setInt(Mp3Directory.TAG_FREQUENCY, frequencyMapping[0][frequency]);
                frequency = frequencyMapping[0][frequency];
            }


            int paddingBit = ((header & 0x00000200) >> 9);

            // Encoding type: Stereo, Joint Stereo, Dual Channel, or Mono
            int mode = ((header & 0x000000C0) >> 6);
            switch (mode){
                case 0:
                    directory.setString(Mp3Directory.TAG_MODE, "Stereo");
                    break;
                case 1:
                    directory.setString(Mp3Directory.TAG_MODE, "Joint stereo");
                    break;
                case 2:
                    directory.setString(Mp3Directory.TAG_MODE, "Dual channel");
                    break;
                case 3:
                    directory.setString(Mp3Directory.TAG_MODE, "Mono");
                    break;
            }

            // Copyright boolean
            int copyright = ((header & 0x00000008) >> 3);
            switch (copyright) {
                case 0:
                    directory.setString(Mp3Directory.TAG_COPYRIGHT, "False");
                    break;
                case 1:
                    directory.setString(Mp3Directory.TAG_COPYRIGHT, "True");
                    break;
            }

            int emphasis = (header & 0x00000003);
            switch (emphasis) {
                case 0:
                    directory.setString(Mp3Directory.TAG_EMPHASIS, "none");
                    break;
                case 1:
                    directory.setString(Mp3Directory.TAG_EMPHASIS, "50/15ms");
                    break;
                case 3:
                    directory.setString(Mp3Directory.TAG_EMPHASIS, "CCITT j.17");
                    break;
            }

            if (bitrate != 0 && bitrate != 15) {
                int frameSize = ((setBitrate(bitrate, layer, id) * 1000) * 144) / frequency;
                directory.setString(Mp3Directory.TAG_FRAME_SIZE, frameSize + " bytes");
            }
        } catch (IOException e) {
            directory.addError(e.getMessage());
        } catch (ImageProcessingException e) {
            directory.addError(e.getMessage());
        }
    }

    private static int setBitrate(int bitrate, int layer, int id)
    {
        int[][] bitrateMapping = new int[14][6];
        bitrateMapping[0] = new int[]{32, 32, 32, 32, 32, 8};
        bitrateMapping[1] = new int[]{64, 48, 40, 64, 48, 16};
        bitrateMapping[2] = new int[]{96, 56, 48, 96, 56, 24};
        bitrateMapping[3] = new int[]{128, 64, 56, 128, 64, 32};
        bitrateMapping[4] = new int[]{160, 80, 64, 160, 80, 64};
        bitrateMapping[5] = new int[]{192, 96, 80, 192, 96, 80};
        bitrateMapping[6] = new int[]{224, 112, 96, 224, 112, 56};
        bitrateMapping[7] = new int[]{256, 128, 112, 256, 128, 64};
        bitrateMapping[8] = new int[]{288, 160, 128, 28, 160, 128};
        bitrateMapping[9] = new int[]{320, 192, 160, 320, 192, 160};
        bitrateMapping[10] = new int[]{352, 224, 192, 352, 224, 112};
        bitrateMapping[11] = new int[]{384, 256, 224, 384, 256, 128};
        bitrateMapping[12] = new int[]{416, 320, 256, 416, 320, 256};
        bitrateMapping[13] = new int[]{448, 384, 320, 448, 384, 320};

        int xPos = 0;
        int yPos = bitrate - 1;

        if (id == 2) {
            // MPEG-2
            switch (layer) {
                case 1:
                    xPos = 5;
                    break;
                case 2:
                    xPos = 4;
                    break;
                case 3:
                    xPos = 3;
                    break;
            }
        } else if (id == 1) {
            // MPEG-1
            switch (layer) {
                case 1:
                    xPos = 2;
                    break;
                case 2:
                    xPos = 1;
                    break;
                case 3:
                    xPos = 0;
                    break;
            }
        }

        return bitrateMapping[yPos][xPos];
    }

/*
    // https://phoxis.org/2010/05/08/synch-safe/
    private static int getSyncSafeSize(int decode)
    {
        int a = decode & 0xFF;
        int b = (decode >> 8) & 0xFF;
        int c  = (decode >> 16) & 0xFF;
        int d = (decode >> 24) & 0xFF;

        int decoded = 0x0;
        decoded = decoded | a;
        decoded = decoded | (b << 7);
        decoded = decoded | (c << 14);
        decoded = decoded | (d << 21);
        return decoded;
    }
*/
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy