oggparsevorbis.c 6.2 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
/**
      Copyright (C) 2005  Michael Ahlberg, Måns Rullgård

      Permission is hereby granted, free of charge, to any person
      obtaining a copy of this software and associated documentation
      files (the "Software"), to deal in the Software without
      restriction, including without limitation the rights to use, copy,
      modify, merge, publish, distribute, sublicense, and/or sell copies
      of the Software, and to permit persons to whom the Software is
      furnished to do so, subject to the following conditions:

      The above copyright notice and this permission notice shall be
      included in all copies or substantial portions of the Software.

      THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
      EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
      MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
      NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
      HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
      WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
      OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
      DEALINGS IN THE SOFTWARE.
**/

#include <stdlib.h>
#include "avformat.h"
#include "bitstream.h"
28
#include "bytestream.h"
29 30
#include "bswap.h"
#include "ogg2.h"
31
#include "avstring.h"
32 33

extern int
M
Måns Rullgård 已提交
34
vorbis_comment(AVFormatContext * as, uint8_t *buf, int size)
35
{
36 37
    uint8_t *p = buf;
    unsigned s, n, j;
38

39
    if (size < 8) /* must have vendor_length and user_comment_list_length */
40 41
        return -1;

M
Måns Rullgård 已提交
42
    s = AV_RL32(p);
43 44 45
    p += 4;
    size -= 4;

46
    if (size - 4 < s)
47 48 49 50 51
        return -1;

    p += s;
    size -= s;

M
Måns Rullgård 已提交
52
    n = AV_RL32(p);
53 54 55
    p += 4;
    size -= 4;

M
Måns Rullgård 已提交
56
    while (size >= 4) {
57 58 59
        char *t, *v;
        int tl, vl;

M
Måns Rullgård 已提交
60
        s = AV_RL32(p);
61 62 63 64 65 66 67 68 69 70 71
        p += 4;
        size -= 4;

        if (size < s)
            break;

        t = p;
        p += s;
        size -= s;
        n--;

M
Måns Rullgård 已提交
72
        v = memchr(t, '=', s);
73 74 75 76 77 78 79
        if (!v)
            continue;

        tl = v - t;
        vl = s - tl - 1;
        v++;

M
Måns Rullgård 已提交
80
        if (tl && vl) {
81 82 83 84
            char tt[tl + 1];
            char ct[vl + 1];

            for (j = 0; j < tl; j++)
M
Måns Rullgård 已提交
85
                tt[j] = toupper(t[j]);
86 87
            tt[tl] = 0;

M
Måns Rullgård 已提交
88
            memcpy(ct, v, vl);
89 90
            ct[vl] = 0;

91
            // took from Vorbis_I_spec
M
Måns Rullgård 已提交
92 93 94 95 96 97 98 99 100 101 102 103
            if (!strcmp(tt, "AUTHOR") || !strcmp(tt, "ARTIST"))
                av_strlcpy(as->author, ct, sizeof(as->author));
            else if (!strcmp(tt, "TITLE"))
                av_strlcpy(as->title, ct, sizeof(as->title));
            else if (!strcmp(tt, "COPYRIGHT"))
                av_strlcpy(as->copyright, ct, sizeof(as->copyright));
            else if (!strcmp(tt, "DESCRIPTION"))
                av_strlcpy(as->comment, ct, sizeof(as->comment));
            else if (!strcmp(tt, "GENRE"))
                av_strlcpy(as->genre, ct, sizeof(as->genre));
            else if (!strcmp(tt, "TRACKNUMBER"))
                as->track = atoi(ct);
M
Måns Rullgård 已提交
104 105
            else if (!strcmp(tt, "ALBUM"))
                av_strlcpy(as->album, ct, sizeof(as->album));
106 107 108 109
        }
    }

    if (size > 0)
M
Måns Rullgård 已提交
110
        av_log(as, AV_LOG_INFO, "%i bytes of comment header remain\n", size);
111
    if (n > 0)
M
Måns Rullgård 已提交
112 113
        av_log(as, AV_LOG_INFO,
               "truncated comment header, %i comments not found\n", n);
114 115 116 117 118 119 120 121 122

    return 0;
}


/** Parse the vorbis header
 * Vorbis Identification header from Vorbis_I_spec.html#vorbis-spec-codec
 * [vorbis_version] = read 32 bits as unsigned integer | Not used
 * [audio_channels] = read 8 bit integer as unsigned | Used
123
 * [audio_sample_rate] = read 32 bits as unsigned integer | Used
124 125 126 127 128 129 130 131
 * [bitrate_maximum] = read 32 bits as signed integer | Not used yet
 * [bitrate_nominal] = read 32 bits as signed integer | Not used yet
 * [bitrate_minimum] = read 32 bits as signed integer | Used as bitrate
 * [blocksize_0] = read 4 bits as unsigned integer | Not Used
 * [blocksize_1] = read 4 bits as unsigned integer | Not Used
 * [framing_flag] = read one bit | Not Used
 *    */

132 133 134 135 136 137 138 139
typedef struct {
    unsigned int len[3];
    unsigned char *packet[3];
} oggvorbis_private_t;


static unsigned int
fixup_vorbis_headers(AVFormatContext * as, oggvorbis_private_t *priv,
M
Måns Rullgård 已提交
140
                     uint8_t **buf)
141 142 143 144 145 146 147 148 149 150 151
{
    int i,offset, len;
    unsigned char *ptr;

    len = priv->len[0] + priv->len[1] + priv->len[2];
    ptr = *buf = av_mallocz(len + len/255 + 64);

    ptr[0] = 2;
    offset = 1;
    offset += av_xiphlacing(&ptr[offset], priv->len[0]);
    offset += av_xiphlacing(&ptr[offset], priv->len[1]);
M
Måns Rullgård 已提交
152
    for (i = 0; i < 3; i++) {
153 154 155 156 157 158 159 160
        memcpy(&ptr[offset], priv->packet[i], priv->len[i]);
        offset += priv->len[i];
    }
    *buf = av_realloc(*buf, offset);
    return offset;
}


161 162 163 164 165 166
static int
vorbis_header (AVFormatContext * s, int idx)
{
    ogg_t *ogg = s->priv_data;
    ogg_stream_t *os = ogg->streams + idx;
    AVStream *st = s->streams[idx];
167
    oggvorbis_private_t *priv;
168 169 170 171

    if (os->seq > 2)
        return 0;

M
Måns Rullgård 已提交
172
    if (os->seq == 0) {
173
        os->private = av_mallocz(sizeof(oggvorbis_private_t));
M
Måns Rullgård 已提交
174
        if (!os->private)
175 176
            return 0;
    }
177

178 179 180
    if (os->psize < 1)
        return -1;

181 182 183 184
    priv = os->private;
    priv->len[os->seq] = os->psize;
    priv->packet[os->seq] = av_mallocz(os->psize);
    memcpy(priv->packet[os->seq], os->buf + os->pstart, os->psize);
185
    if (os->buf[os->pstart] == 1) {
M
Måns Rullgård 已提交
186
        uint8_t *p = os->buf + os->pstart + 11; //skip up to the audio channels
187 188 189 190

        if (os->psize != 30)
            return -1;

191 192 193 194
        st->codec->channels = bytestream_get_byte(&p);
        st->codec->sample_rate = bytestream_get_le32(&p);
        p += 4; // skip maximum bitrate
        st->codec->bit_rate = bytestream_get_le32(&p); // nominal bitrate
195

196 197
        st->codec->codec_type = CODEC_TYPE_AUDIO;
        st->codec->codec_id = CODEC_ID_VORBIS;
198

M
Måns Rullgård 已提交
199 200
        st->time_base.num = 1;
        st->time_base.den = st->codec->sample_rate;
201
    } else if (os->buf[os->pstart] == 3) {
202 203
        if (os->psize > 8)
            vorbis_comment (s, os->buf + os->pstart + 7, os->psize - 8);
204
    } else {
205 206
        st->codec->extradata_size =
            fixup_vorbis_headers(s, priv, &st->codec->extradata);
207 208 209 210 211 212 213 214 215 216
    }

    return os->seq < 3;
}

ogg_codec_t vorbis_codec = {
    .magic = "\001vorbis",
    .magicsize = 7,
    .header = vorbis_header
};