rtpenc_mpv.c 3.63 KB
Newer Older
1 2
/*
 * RTP packetization for MPEG video
3 4
 * Copyright (c) 2002 Fabrice Bellard
 * Copyright (c) 2007 Luca Abeni
5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
 *
 * This file is part of FFmpeg.
 *
 * FFmpeg is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation; either
 * version 2.1 of the License, or (at your option) any later version.
 *
 * FFmpeg is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with FFmpeg; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
 */
22

23
#include "libavcodec/internal.h"
24
#include "avformat.h"
25
#include "rtpenc.h"
26 27 28 29 30

/* NOTE: a single frame must be passed with sequence header if
   needed. XXX: use slices. */
void ff_rtp_send_mpegvideo(AVFormatContext *s1, const uint8_t *buf1, int size)
{
31
    RTPMuxContext *s = s1->priv_data;
32 33
    int len, h, max_packet_size;
    uint8_t *q;
34
    const uint8_t *end = buf1 + size;
35
    int begin_of_slice, end_of_slice, frame_type, temporal_reference;
36 37

    max_packet_size = s->max_payload_size;
38 39
    begin_of_slice = 1;
    end_of_slice = 0;
40
    frame_type = 0;
41
    temporal_reference = 0;
42 43

    while (size > 0) {
44 45 46
        int begin_of_sequence;

        begin_of_sequence = 0;
47 48 49 50
        len = max_packet_size - 4;

        if (len >= size) {
            len = size;
51 52 53 54 55 56 57 58
            end_of_slice = 1;
        } else {
            const uint8_t *r, *r1;
            int start_code;

            r1 = buf1;
            while (1) {
                start_code = -1;
59
                r = avpriv_find_start_code(r1, end, &start_code);
60 61
                if((start_code & 0xFFFFFF00) == 0x100) {
                    /* New start code found */
62 63
                    if (start_code == 0x100) {
                        frame_type = (r[1] & 0x38) >> 3;
64
                        temporal_reference = (int)r[0] << 2 | r[1] >> 6;
65
                    }
66 67 68
                    if (start_code == 0x1B8) {
                        begin_of_sequence = 1;
                    }
69

70
                    if (r - buf1 - 4 <= len) {
71 72 73 74 75 76 77 78 79
                        /* The current slice fits in the packet */
                        if (begin_of_slice == 0) {
                            /* no slice at the beginning of the packet... */
                            end_of_slice = 1;
                            len = r - buf1 - 4;
                            break;
                        }
                        r1 = r;
                    } else {
80
                        if ((r1 - buf1 > 4) && (r - r1 < max_packet_size)) {
81 82 83 84 85 86 87 88 89
                            len = r1 - buf1 - 4;
                            end_of_slice = 1;
                        }
                        break;
                    }
                } else {
                    break;
                }
            }
90 91
        }

92
        h = 0;
93
        h |= temporal_reference << 16;
94
        h |= begin_of_sequence << 13;
95 96
        h |= begin_of_slice << 12;
        h |= end_of_slice << 11;
97
        h |= frame_type << 8;
98

99 100 101 102 103 104 105 106 107
        q = s->buf;
        *q++ = h >> 24;
        *q++ = h >> 16;
        *q++ = h >> 8;
        *q++ = h;

        memcpy(q, buf1, len);
        q += len;

Diego Biurrun's avatar
Diego Biurrun committed
108
        /* 90kHz time stamp */
109
        s->timestamp = s->cur_timestamp;
110 111 112 113
        ff_rtp_send_data(s1, s->buf, q - s->buf, (len == size));

        buf1 += len;
        size -= len;
114 115
        begin_of_slice = end_of_slice;
        end_of_slice = 0;
116 117
    }
}