Older/MediaServer/libmpeg/source/mpeg-ps-enc.c

296 lines
8.5 KiB
C
Raw Permalink Normal View History

2024-10-01 00:12:57 +08:00
// ITU-T H.222.0(06/2012)
// Information technology - Generic coding of moving pictures and associated audio information: Systems
// 2.5.3.1 Program stream(p74)
#include "mpeg-pes-internal.h"
#include "mpeg-ps-internal.h"
#include "mpeg-util.h"
#include "mpeg-ps.h"
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include <time.h>
#define MAX_PES_HEADER 1024 // pack_header + system_header + psm
#define MAX_PES_PACKET 0xFF80 // 64k pes data, reserved 0x7F for hik
struct ps_muxer_t
{
struct psm_t psm;
struct ps_pack_header_t pack;
struct ps_system_header_t system;
int h26x_with_aud;
unsigned int psm_period;
unsigned int scr_period;
struct ps_muxer_func_t func;
void* param;
// uint8_t packet[MAX_PACKET_SIZE];
};
static struct pes_t* ps_stream_find(struct ps_muxer_t *ps, int streamid)
{
size_t i;
for (i = 0; i < ps->psm.stream_count; i++)
{
if (streamid == ps->psm.streams[i].sid)
return &ps->psm.streams[i];
}
return NULL;
}
int ps_muxer_input(struct ps_muxer_t* ps, int streamid, int flags, int64_t pts, int64_t dts, const void* data, size_t bytes)
{
int r, first;
size_t i, n, sz;
uint8_t *packet;
struct pes_t* stream;
const uint8_t* payload;
i = 0;
first = 1;
payload = (const uint8_t*)data;
stream = ps_stream_find(ps, streamid);
if (NULL == stream) return -1; // not found
stream->data_alignment_indicator = (flags & MPEG_FLAG_IDR_FRAME) ? 1 : 0; // idr frame
stream->pts = pts;
stream->dts = dts;
// Add PSM for IDR frame
ps->psm_period = ((flags & MPEG_FLAG_IDR_FRAME) && mpeg_stream_type_video(stream->codecid)) ? 0 : ps->psm_period;
ps->h26x_with_aud = (flags & MPEG_FLAG_H264_H265_WITH_AUD) ? 1 : 0;
// TODO:
// 1. update packet header program_mux_rate
// 2. update system header rate_bound
// alloc once (include Multi-PES packet)
sz = bytes + MAX_PES_HEADER + (bytes/MAX_PES_PACKET+1) * 64; // 64 = 0x000001 + stream_id + PES_packet_length + other
packet = ps->func.alloc(ps->param, sz);
if(!packet) return -ENOMEM;
// write pack_header(p74)
// 2.7.1 Frequency of coding the system clock reference
// http://www.bretl.com/mpeghtml/SCR.HTM
//the maximum allowed interval between SCRs is 700ms
//ps->pack.system_clock_reference_base = (dts-3600) % (((int64_t)1)<<33);
ps->pack.system_clock_reference_base = dts >= 3600 ? (dts - 3600) : 0;
ps->pack.system_clock_reference_extension = 0;
ps->pack.program_mux_rate = 6106;
i += pack_header_write(&ps->pack, packet + i);
#if !defined(MPEG_FIX_VLC_3_X_PS_SYSTEM_HEADER)
// https://github.com/videolan/vlc/blob/3.0.x/modules/demux/mpeg/ps.h#L488
// fix ps_pkt_parse_system -> ps_track_fill with default mp1/2 audio codec(without psm)
// write system_header(p76)
if(0 == (ps->psm_period % 30))
i += system_header_write(&ps->system, packet + i);
#endif
// write program_stream_map(p79)
if (0 == (ps->psm_period % 30))
{
#if defined(MPEG_CLOCK_EXTENSION_DESCRIPTOR)
ps->psm.clock = time() * 1000; // todo: gettimeofday
#endif
i += psm_write(&ps->psm, packet + i);
}
// check packet size
assert(i < MAX_PES_HEADER);
// write data
while(bytes > 0)
{
uint8_t *p;
uint8_t *pes = packet + i;
p = pes + pes_write_header(stream, pes, sz - i);
stream->pts = stream->dts = PTS_NO_VALUE; // clear pts/dts flags
stream->data_alignment_indicator = 0; // clear flags
assert(p - pes < 64);
if(first)
{
if (PSI_STREAM_H264 == stream->codecid && !ps->h26x_with_aud)
{
// 2.14 Carriage of Rec. ITU-T H.264 | ISO/IEC 14496-10 video
// Each AVC access unit shall contain an access unit delimiter NAL Unit
nbo_w32(p, 0x00000001);
p[4] = 0x09; // AUD
p[5] = 0xE0; // any slice type (0xe) + rbsp stop one bit
p += 6;
}
else if (PSI_STREAM_H265 == stream->codecid && !ps->h26x_with_aud)
{
// 2.17 Carriage of HEVC
// Each HEVC access unit shall contain an access unit delimiter NAL unit.
nbo_w32(p, 0x00000001);
p[4] = 0x46; // 35-AUD_NUT
p[5] = 0x01;
p[6] = 0x50; // B&P&I (0x2) + rbsp stop one bit
p += 7;
}
else if (PSI_STREAM_H266 == stream->codecid && !ps->h26x_with_aud)
{
// 2.23 Carriage of VVC
// Each VVC access unit shall contain an access unit delimiter NAL unit
nbo_w32(p, 0x00000001);
p[4] = 0x00; // 20-AUD_NUT
p[5] = 0xA1;
p[6] = 0x18; // B&P&I (0x1) + rbsp stop one bit
p += 7;
}
}
// PES_packet_length = PES-Header + Payload-Size
// A value of 0 indicates that the PES packet length is neither specified nor bounded
// and is allowed only in PES packets whose payload consists of bytes from a
// video elementary stream contained in transport stream packets
if((p - pes - 6) + bytes > MAX_PES_PACKET)
{
nbo_w16(pes + 4, MAX_PES_PACKET);
n = MAX_PES_PACKET - (p - pes - 6);
}
else
{
nbo_w16(pes + 4, (uint16_t)((p - pes - 6) + bytes));
n = bytes;
}
memcpy(p, payload, n);
payload += n;
bytes -= n;
// notify packet already
i += n + (p - pes);
// i = 0; // clear value, the next pes packet don't need pack_header
first = 0; // clear first packet flag
pts = dts = 0; // only first packet write PTS/DTS
}
assert(i < sz);
r = ps->func.write(ps->param, stream->sid, packet, i);
ps->func.free(ps->param, packet);
++ps->psm_period;
return r;
}
struct ps_muxer_t* ps_muxer_create(const struct ps_muxer_func_t *func, void* param)
{
struct ps_muxer_t *ps = NULL;
assert(func);
ps = (struct ps_muxer_t *)calloc(1, sizeof(struct ps_muxer_t));
if(!ps)
return NULL;
memcpy(&ps->func, func, sizeof(ps->func));
ps->param = param;
ps->system.rate_bound = 26234; //10493600~10mbps(50BPS * 8 = 400bps)
// ps->system.audio_bound = 1; // [0,32] max active audio streams
// ps->system.video_bound = 1; // [0,16] max active video streams
ps->system.fixed_flag = 0; // 1-fixed bitrate, 0-variable bitrate
ps->system.CSPS_flag = 0; // meets the constraints defined in 2.7.9.
ps->system.packet_rate_restriction_flag = 0; // dependence CSPS_flag
ps->system.system_audio_lock_flag = 0; // all audio stream sampling rate is constant
ps->system.system_video_lock_flag = 0; // all video stream frequency is constant
//ps->psm.ver = 1;
//ps->psm.stream_count = 2;
//ps->psm.streams[0].element_stream_id = PES_SID_VIDEO;
//ps->psm.streams[0].stream_type = PSI_STREAM_H264;
//ps->psm.streams[1].element_stream_id = PES_SID_AUDIO;
//ps->psm.streams[1].stream_type = PSI_STREAM_AAC;
return ps;
}
int ps_muxer_destroy(struct ps_muxer_t* ps)
{
size_t i;
for (i = 0; i < ps->psm.stream_count; i++)
{
if (ps->psm.streams[i].esinfo)
{
free(ps->psm.streams[i].esinfo);
ps->psm.streams[i].esinfo = NULL;
}
}
free(ps);
return 0;
}
int ps_muxer_add_stream(struct ps_muxer_t* ps, int codecid, const void* extradata, size_t bytes)
{
struct psm_t* psm;
struct pes_t* pes;
assert(bytes < 512);
if (!ps || ps->psm.stream_count >= sizeof(ps->psm.streams) / sizeof(ps->psm.streams[0]))
{
assert(0);
return -1;
}
psm = &ps->psm;
pes = &psm->streams[psm->stream_count];
if (mpeg_stream_type_video(codecid))
{
pes->sid = (uint8_t)(PES_SID_VIDEO + ps->system.video_bound);
assert(ps->system.video_bound + 1 < 16);
++ps->system.video_bound; // [0,16] max active video streams
ps->system.streams[ps->system.stream_count].buffer_bound_scale = 1;
/* FIXME -- VCD uses 46, SVCD uses 230, ffmpeg has 230 with a note that it is small */
ps->system.streams[ps->system.stream_count].buffer_size_bound = 400 /* 8191-13 bits max value */;
}
else if (mpeg_stream_type_audio(codecid))
{
pes->sid = (uint8_t)(PES_SID_AUDIO + ps->system.audio_bound);
assert(ps->system.audio_bound + 1 < 32);
++ps->system.audio_bound; // [0,32] max active audio streams
ps->system.streams[ps->system.stream_count].buffer_bound_scale = 0;
/* This value HAS to be used for VCD (see VCD standard, p. IV-7).
* Right now it is also used for everything else. */
ps->system.streams[ps->system.stream_count].buffer_size_bound = 32 /* 4 * 1024 / 128 */;
}
else
{
assert(0);
return -1;
}
if (bytes > 0)
{
pes->esinfo = (uint8_t*)malloc(bytes);
if (!pes->esinfo)
return -1;
memcpy(pes->esinfo, extradata, bytes);
pes->esinfo_len = (uint16_t)bytes;
}
assert(psm->stream_count == ps->system.stream_count);
ps->system.streams[ps->system.stream_count].stream_id = pes->sid;
++ps->system.stream_count;
pes->codecid = (uint8_t)codecid;
++psm->stream_count;
++psm->ver;
ps->psm_period = 0; // immediate update psm
return pes->sid;
}