aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorFabrice Bellard <fabrice@bellard.org>2003-09-16 12:56:42 +0000
committerFabrice Bellard <fabrice@bellard.org>2003-09-16 12:56:42 +0000
commit69ef9450d8101dc30c6a2ba9e1c88af966f928f0 (patch)
tree1bd3e7e4d0a0bea68749a16d3554a4a4919355dc
parent05fbd0a2b037cbe99b7f2868922c8426acc44989 (diff)
downloadffmpeg-69ef9450d8101dc30c6a2ba9e1c88af966f928f0.tar.gz
generate correct PTS in transport stream - use mpeg2 encoder by default
Originally committed as revision 2282 to svn://svn.ffmpeg.org/ffmpeg/trunk
-rw-r--r--libavformat/mpegtsenc.c174
1 files changed, 101 insertions, 73 deletions
diff --git a/libavformat/mpegtsenc.c b/libavformat/mpegtsenc.c
index 9b1d9d36b4..e4d7f3d7b8 100644
--- a/libavformat/mpegtsenc.c
+++ b/libavformat/mpegtsenc.c
@@ -185,6 +185,7 @@ int mpegts_write_section1(MpegTSSection *s, int tid, int id,
/* a PES packet header is generated every DEFAULT_PES_HEADER_FREQ packets */
#define DEFAULT_PES_HEADER_FREQ 16
+#define DEFAULT_PES_PAYLOAD_SIZE ((DEFAULT_PES_HEADER_FREQ - 1) * 184 + 170)
/* we retransmit the SI info at this rate */
#define SDT_RETRANS_TIME 500
@@ -193,9 +194,9 @@ int mpegts_write_section1(MpegTSSection *s, int tid, int id,
typedef struct MpegTSWriteStream {
int pid; /* stream associated pid */
int cc;
- int packet_index;
- int pes_packet_count;
- uint8_t packet[TS_PACKET_SIZE];
+ int payload_index;
+ int64_t payload_pts;
+ uint8_t payload[DEFAULT_PES_PAYLOAD_SIZE];
} MpegTSWriteStream;
typedef struct MpegTSService {
@@ -398,6 +399,7 @@ static int mpegts_write_header(AVFormatContext *s)
goto fail;
st->priv_data = ts_st;
ts_st->pid = DEFAULT_START_PID + i;
+ ts_st->payload_pts = AV_NOPTS_VALUE;
/* update PCR pid if needed */
if (st->codec.codec_type == CODEC_TYPE_VIDEO &&
service->pcr_pid == 0x1fff)
@@ -453,82 +455,110 @@ static void retransmit_si_info(AVFormatContext *s)
}
}
-static int mpegts_write_packet(AVFormatContext *s, int stream_index,
- const uint8_t *buf, int size, int64_t pts1)
+/* NOTE: pes_data contains all the PES packet */
+static void mpegts_write_pes(AVFormatContext *s, AVStream *st,
+ const uint8_t *payload, int payload_size,
+ int64_t pts)
{
- AVStream *st = s->streams[stream_index];
MpegTSWriteStream *ts_st = st->priv_data;
+ uint8_t buf[TS_PACKET_SIZE];
uint8_t *q;
- int val, write_pts, is_start, len;
- int64_t pts;
+ int val, is_start, len, ts_len, header_len;
- while (size > 0) {
- if (ts_st->packet_index == 0) {
- retransmit_si_info(s);
+ is_start = 1;
+ while (payload_size > 0) {
+ retransmit_si_info(s);
- /* new PES header ? */
- is_start = 0;
- if (++ts_st->pes_packet_count == DEFAULT_PES_HEADER_FREQ) {
- ts_st->pes_packet_count = 0;
- is_start = 1;
- }
- /* prepare packet header */
- q = ts_st->packet;
- *q++ = 0x47;
- val = (ts_st->pid >> 8);
- if (is_start)
- val |= 0x40;
- *q++ = val;
- *q++ = ts_st->pid;
- *q++ = 0x10 | ts_st->cc;
- ts_st->cc = (ts_st->cc + 1) & 0xf;
- if (is_start) {
- /* write PES header */
+ /* prepare packet header */
+ q = buf;
+ *q++ = 0x47;
+ val = (ts_st->pid >> 8);
+ if (is_start)
+ val |= 0x40;
+ *q++ = val;
+ *q++ = ts_st->pid;
+ *q++ = 0x10 | ts_st->cc;
+ ts_st->cc = (ts_st->cc + 1) & 0xf;
+ if (is_start) {
+ /* write PES header */
+ *q++ = 0x00;
+ *q++ = 0x00;
+ *q++ = 0x01;
+ if (st->codec.codec_type == CODEC_TYPE_VIDEO)
+ *q++ = 0xe0;
+ else
+ *q++ = 0xc0;
+ if (pts != AV_NOPTS_VALUE)
+ header_len = 8;
+ else
+ header_len = 3;
+ len = payload_size + header_len;
+ *q++ = len >> 8;
+ *q++ = len;
+ *q++ = 0x80;
+ if (pts != AV_NOPTS_VALUE) {
+ *q++ = 0x80; /* PTS only */
+ *q++ = 0x05; /* header len */
+ val = (0x02 << 4) |
+ (((pts >> 30) & 0x07) << 1) | 1;
+ *q++ = val;
+ val = (((pts >> 15) & 0x7fff) << 1) | 1;
+ *q++ = val >> 8;
+ *q++ = val;
+ val = (((pts) & 0x7fff) << 1) | 1;
+ *q++ = val >> 8;
+ *q++ = val;
+ } else {
*q++ = 0x00;
*q++ = 0x00;
- *q++ = 0x01;
- if (st->codec.codec_type == CODEC_TYPE_VIDEO)
- *q++ = 0xe0;
- else
- *q++ = 0xc0;
- *q++ = 0; /* unbounded size */
- *q++ = 0;
- *q++ = 0x80;
- write_pts = 0; /* XXX: enable it */
- if (write_pts) {
- *q++ = 0x80; /* PTS only */
- *q++ = 0x05; /* header len */
- pts = pts1;
- val = (0x02 << 4) |
- (((pts >> 30) & 0x07) << 1) | 1;
- *q++ = val;
- val = (((pts >> 15) & 0x7fff) << 1) | 1;
- *q++ = val >> 8;
- *q++ = val;
- val = (((pts) & 0x7fff) << 1) | 1;
- *q++ = val >> 8;
- *q++ = val;
- } else {
- *q++ = 0x00;
- *q++ = 0x00;
- }
}
- ts_st->packet_index = q - ts_st->packet;
+ is_start = 0;
}
- len = TS_PACKET_SIZE - ts_st->packet_index;
- if (len == 0) {
- put_buffer(&s->pb, ts_st->packet, TS_PACKET_SIZE);
- ts_st->packet_index = 0;
- } else {
- if (len > size)
- len = size;
- memcpy(ts_st->packet + ts_st->packet_index, buf, len);
- size -= len;
- buf += len;
- ts_st->packet_index += len;
+ /* write header */
+ ts_len = q - buf;
+ put_buffer(&s->pb, buf, ts_len);
+ /* write data */
+ len = TS_PACKET_SIZE - ts_len;
+ if (len > payload_size)
+ len = payload_size;
+ put_buffer(&s->pb, payload, len);
+ payload += len;
+ payload_size -= len;
+ ts_len += len;
+ /* stuffing */
+ len = TS_PACKET_SIZE - ts_len;
+ if (len > 0) {
+ memset(buf, 0xff, len);
+ put_buffer(&s->pb, buf, len);
}
}
put_flush_packet(&s->pb);
+}
+
+static int mpegts_write_packet(AVFormatContext *s, int stream_index,
+ const uint8_t *buf, int size, int64_t pts1)
+{
+ AVStream *st = s->streams[stream_index];
+ MpegTSWriteStream *ts_st = st->priv_data;
+ int len;
+
+ while (size > 0) {
+ len = DEFAULT_PES_PAYLOAD_SIZE - ts_st->payload_index;
+ if (len > size)
+ len = size;
+ memcpy(ts_st->payload + ts_st->payload_index, buf, len);
+ buf += len;
+ size -= len;
+ ts_st->payload_index += len;
+ if (ts_st->payload_pts == AV_NOPTS_VALUE)
+ ts_st->payload_pts = pts1;
+ if (ts_st->payload_index >= DEFAULT_PES_PAYLOAD_SIZE) {
+ mpegts_write_pes(s, st, ts_st->payload, ts_st->payload_index,
+ ts_st->payload_pts);
+ ts_st->payload_pts = AV_NOPTS_VALUE;
+ ts_st->payload_index = 0;
+ }
+ }
return 0;
}
@@ -544,11 +574,9 @@ static int mpegts_write_end(AVFormatContext *s)
for(i = 0; i < s->nb_streams; i++) {
st = s->streams[i];
ts_st = st->priv_data;
- if (ts_st->packet_index != 0) {
- /* put a known value at the end */
- memset(ts_st->packet + ts_st->packet_index, 0xff,
- TS_PACKET_SIZE - ts_st->packet_index);
- put_buffer(&s->pb, ts_st->packet, TS_PACKET_SIZE);
+ if (ts_st->payload_index > 0) {
+ mpegts_write_pes(s, st, ts_st->payload, ts_st->payload_index,
+ ts_st->payload_pts);
}
}
put_flush_packet(&s->pb);
@@ -575,7 +603,7 @@ AVOutputFormat mpegts_mux = {
"ts",
sizeof(MpegTSWrite),
CODEC_ID_MP2,
- CODEC_ID_MPEG1VIDEO,
+ CODEC_ID_MPEG2VIDEO,
mpegts_write_header,
mpegts_write_packet,
mpegts_write_end,