1cabdff1aSopenharmony_ci/* 2cabdff1aSopenharmony_ci * RTP packetizer for VC-2 HQ payload format (draft version 1) - experimental 3cabdff1aSopenharmony_ci * Copyright (c) 2016 Thomas Volkert <thomas@netzeal.de> 4cabdff1aSopenharmony_ci * 5cabdff1aSopenharmony_ci * This file is part of FFmpeg. 6cabdff1aSopenharmony_ci * 7cabdff1aSopenharmony_ci * FFmpeg is free software; you can redistribute it and/or 8cabdff1aSopenharmony_ci * modify it under the terms of the GNU Lesser General Public 9cabdff1aSopenharmony_ci * License as published by the Free Software Foundation; either 10cabdff1aSopenharmony_ci * version 2.1 of the License, or (at your option) any later version. 11cabdff1aSopenharmony_ci * 12cabdff1aSopenharmony_ci * FFmpeg is distributed in the hope that it will be useful, 13cabdff1aSopenharmony_ci * but WITHOUT ANY WARRANTY; without even the implied warranty of 14cabdff1aSopenharmony_ci * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 15cabdff1aSopenharmony_ci * Lesser General Public License for more details. 16cabdff1aSopenharmony_ci * 17cabdff1aSopenharmony_ci * You should have received a copy of the GNU Lesser General Public 18cabdff1aSopenharmony_ci * License along with FFmpeg; if not, write to the Free Software 19cabdff1aSopenharmony_ci * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA 20cabdff1aSopenharmony_ci */ 21cabdff1aSopenharmony_ci 22cabdff1aSopenharmony_ci#include "libavutil/intreadwrite.h" 23cabdff1aSopenharmony_ci#include "libavcodec/dirac.h" 24cabdff1aSopenharmony_ci#include "libavcodec/get_bits.h" 25cabdff1aSopenharmony_ci#include "libavcodec/golomb.h" 26cabdff1aSopenharmony_ci 27cabdff1aSopenharmony_ci#include "avformat.h" 28cabdff1aSopenharmony_ci#include "rtpenc.h" 29cabdff1aSopenharmony_ci 30cabdff1aSopenharmony_ci#define RTP_VC2HQ_PL_HEADER_SIZE 4 31cabdff1aSopenharmony_ci 32cabdff1aSopenharmony_ci#define DIRAC_DATA_UNIT_HEADER_SIZE 13 33cabdff1aSopenharmony_ci#define DIRAC_PIC_NR_SIZE 4 34cabdff1aSopenharmony_ci#define DIRAC_RTP_PCODE_HQ_PIC_FRAGMENT 0xEC 35cabdff1aSopenharmony_ci 36cabdff1aSopenharmony_cistatic void send_packet(AVFormatContext *ctx, uint8_t parse_code, int info_hdr_size, const uint8_t *buf, int size, int i, int f, int rtp_m) 37cabdff1aSopenharmony_ci{ 38cabdff1aSopenharmony_ci RTPMuxContext *rtp_ctx = ctx->priv_data; 39cabdff1aSopenharmony_ci 40cabdff1aSopenharmony_ci AV_WB16(&rtp_ctx->buf[0], 0); /* extended sequence number */ 41cabdff1aSopenharmony_ci AV_WB8 (&rtp_ctx->buf[2], i ? (f ? (0x03) : (0x02)) : 0x00); /* flags: interlaced, second field */ 42cabdff1aSopenharmony_ci AV_WB8 (&rtp_ctx->buf[3], parse_code); 43cabdff1aSopenharmony_ci if (size > 0) 44cabdff1aSopenharmony_ci memcpy(&rtp_ctx->buf[4 + info_hdr_size], buf, size); 45cabdff1aSopenharmony_ci ff_rtp_send_data(ctx, rtp_ctx->buf, RTP_VC2HQ_PL_HEADER_SIZE + info_hdr_size + size, rtp_m); 46cabdff1aSopenharmony_ci} 47cabdff1aSopenharmony_ci 48cabdff1aSopenharmony_cistatic void send_picture(AVFormatContext *ctx, const uint8_t *buf, int size, int interlaced) 49cabdff1aSopenharmony_ci{ 50cabdff1aSopenharmony_ci RTPMuxContext *rtp_ctx = ctx->priv_data; 51cabdff1aSopenharmony_ci GetBitContext gc; 52cabdff1aSopenharmony_ci int lvl, second_field; 53cabdff1aSopenharmony_ci uint32_t pic_nr, wavelet_depth, prefix_bytes, size_scaler; 54cabdff1aSopenharmony_ci uint16_t frag_len; 55cabdff1aSopenharmony_ci char *info_hdr = &rtp_ctx->buf[4]; 56cabdff1aSopenharmony_ci 57cabdff1aSopenharmony_ci pic_nr = AV_RB32(&buf[0]); 58cabdff1aSopenharmony_ci buf += DIRAC_PIC_NR_SIZE; 59cabdff1aSopenharmony_ci size -= DIRAC_PIC_NR_SIZE; 60cabdff1aSopenharmony_ci second_field = interlaced && (pic_nr & 0x01); 61cabdff1aSopenharmony_ci 62cabdff1aSopenharmony_ci init_get_bits(&gc, buf, 8 * size); 63cabdff1aSopenharmony_ci get_interleaved_ue_golomb(&gc); /* wavelet_idx */ 64cabdff1aSopenharmony_ci wavelet_depth = get_interleaved_ue_golomb(&gc); 65cabdff1aSopenharmony_ci get_interleaved_ue_golomb(&gc); /* num_x */ 66cabdff1aSopenharmony_ci get_interleaved_ue_golomb(&gc); /* num_y */ 67cabdff1aSopenharmony_ci prefix_bytes = get_interleaved_ue_golomb(&gc); 68cabdff1aSopenharmony_ci size_scaler = get_interleaved_ue_golomb(&gc); 69cabdff1aSopenharmony_ci /* pass the quantization matrices */ 70cabdff1aSopenharmony_ci get_interleaved_ue_golomb(&gc); 71cabdff1aSopenharmony_ci for(lvl = 0; lvl < wavelet_depth; lvl++) 72cabdff1aSopenharmony_ci { 73cabdff1aSopenharmony_ci get_interleaved_ue_golomb(&gc); 74cabdff1aSopenharmony_ci get_interleaved_ue_golomb(&gc); 75cabdff1aSopenharmony_ci get_interleaved_ue_golomb(&gc); 76cabdff1aSopenharmony_ci } 77cabdff1aSopenharmony_ci 78cabdff1aSopenharmony_ci frag_len = (get_bits_count(&gc) + 7) / 8; /* length of transform parameters */ 79cabdff1aSopenharmony_ci 80cabdff1aSopenharmony_ci AV_WB32(&info_hdr[ 0], pic_nr); 81cabdff1aSopenharmony_ci AV_WB16(&info_hdr[ 4], prefix_bytes); 82cabdff1aSopenharmony_ci AV_WB16(&info_hdr[ 6], size_scaler); 83cabdff1aSopenharmony_ci AV_WB16(&info_hdr[ 8], frag_len); 84cabdff1aSopenharmony_ci AV_WB16(&info_hdr[10], 0 /* nr. of slices */); 85cabdff1aSopenharmony_ci send_packet(ctx, DIRAC_RTP_PCODE_HQ_PIC_FRAGMENT, 12, buf, frag_len, interlaced, second_field, 0); 86cabdff1aSopenharmony_ci buf += frag_len; 87cabdff1aSopenharmony_ci size -= frag_len; 88cabdff1aSopenharmony_ci 89cabdff1aSopenharmony_ci while (size > 0) { 90cabdff1aSopenharmony_ci frag_len = FFMIN(rtp_ctx->max_payload_size - 20 /* pl header */, size); 91cabdff1aSopenharmony_ci AV_WB16(&info_hdr[ 8], frag_len); 92cabdff1aSopenharmony_ci AV_WB16(&info_hdr[10], 1 /* nr. of slices */); 93cabdff1aSopenharmony_ci AV_WB16(&info_hdr[12], 0 /* slice x */); 94cabdff1aSopenharmony_ci AV_WB16(&info_hdr[14], 0 /* slice y */); 95cabdff1aSopenharmony_ci 96cabdff1aSopenharmony_ci size -= frag_len; 97cabdff1aSopenharmony_ci send_packet(ctx, DIRAC_RTP_PCODE_HQ_PIC_FRAGMENT, 16, buf, frag_len, interlaced, second_field, size > 0 ? 0 : 1); 98cabdff1aSopenharmony_ci buf += frag_len; 99cabdff1aSopenharmony_ci } 100cabdff1aSopenharmony_ci} 101cabdff1aSopenharmony_ci 102cabdff1aSopenharmony_civoid ff_rtp_send_vc2hq(AVFormatContext *ctx, const uint8_t *frame_buf, int frame_size, int interlaced) 103cabdff1aSopenharmony_ci{ 104cabdff1aSopenharmony_ci const uint8_t *end = frame_buf + frame_size; 105cabdff1aSopenharmony_ci const uint8_t *unit = frame_buf; 106cabdff1aSopenharmony_ci uint8_t parse_code; 107cabdff1aSopenharmony_ci uint32_t unit_size; 108cabdff1aSopenharmony_ci 109cabdff1aSopenharmony_ci while (unit < end) { 110cabdff1aSopenharmony_ci parse_code = unit[4]; 111cabdff1aSopenharmony_ci unit_size = AV_RB32(&unit[5]); 112cabdff1aSopenharmony_ci 113cabdff1aSopenharmony_ci switch (parse_code) { 114cabdff1aSopenharmony_ci /* sequence header */ 115cabdff1aSopenharmony_ci /* end of sequence */ 116cabdff1aSopenharmony_ci case DIRAC_PCODE_SEQ_HEADER: 117cabdff1aSopenharmony_ci case DIRAC_PCODE_END_SEQ: 118cabdff1aSopenharmony_ci send_packet(ctx, parse_code, 0, unit + DIRAC_DATA_UNIT_HEADER_SIZE, unit_size - DIRAC_DATA_UNIT_HEADER_SIZE, 0, 0, 0); 119cabdff1aSopenharmony_ci break; 120cabdff1aSopenharmony_ci /* HQ picture */ 121cabdff1aSopenharmony_ci case DIRAC_PCODE_PICTURE_HQ: 122cabdff1aSopenharmony_ci send_picture(ctx, unit + DIRAC_DATA_UNIT_HEADER_SIZE, unit_size - DIRAC_DATA_UNIT_HEADER_SIZE, interlaced); 123cabdff1aSopenharmony_ci break; 124cabdff1aSopenharmony_ci /* parse codes without specification */ 125cabdff1aSopenharmony_ci case DIRAC_PCODE_AUX: 126cabdff1aSopenharmony_ci case DIRAC_PCODE_PAD: 127cabdff1aSopenharmony_ci break; 128cabdff1aSopenharmony_ci default: 129cabdff1aSopenharmony_ci avpriv_report_missing_feature(ctx, "VC-2 parse code %d", parse_code); 130cabdff1aSopenharmony_ci break; 131cabdff1aSopenharmony_ci } 132cabdff1aSopenharmony_ci unit += unit_size; 133cabdff1aSopenharmony_ci } 134cabdff1aSopenharmony_ci} 135