Files
media-server/libmpeg/test/mpeg-ts-multi-program-test.cpp
2022-10-29 13:04:43 +08:00

158 lines
4.7 KiB
C++

#include "mpeg-ps.h"
#include "mpeg-ts.h"
#include "mov-reader.h"
#include "mov-format.h"
#include "mpeg4-aac.h"
#include "mpeg4-avc.h"
#include "mpeg4-hevc.h"
#include <assert.h>
#include <string.h>
#include <stdio.h>
#include <map>
extern "C" const struct mov_buffer_t* mov_file_buffer(void);
static uint8_t s_packet[2 * 1024 * 1024];
static uint8_t s_buffer[4 * 1024 * 1024];
struct mpeg_ts_multi_program_test_t
{
int count;
struct
{
int pn;
int track;
int stream;
uint8_t object;
union
{
mpeg4_avc_t avc;
mpeg4_hevc_t hevc;
} v;
union
{
mpeg4_aac_t aac;
} a;
} pn[4];
void* ts;
};
static void* ts_alloc(void* /*param*/, size_t bytes)
{
static char s_buffer[188];
assert(bytes <= sizeof(s_buffer));
return s_buffer;
}
static void ts_free(void* /*param*/, void* /*packet*/)
{
return;
}
static int ts_write(void* param, const void* packet, size_t bytes)
{
return 1 == fwrite(packet, bytes, 1, (FILE*)param) ? 0 : ferror((FILE*)param);
}
static void onread(void* param, uint32_t track, const void* data, size_t bytes, int64_t pts, int64_t dts, int flags)
{
mpeg_ts_multi_program_test_t* ctx = (mpeg_ts_multi_program_test_t*)param;
for (int i = 0; i < ctx->count; i++)
{
if (ctx->pn[i].track == track)
{
if (MOV_OBJECT_H264 == ctx->pn[i].object)
{
bytes = h264_mp4toannexb(&ctx->pn[i].v.avc, data, bytes, s_packet, sizeof(s_packet));
data = s_packet;
}
else if (MOV_OBJECT_HEVC == ctx->pn[i].object)
{
bytes = h265_mp4toannexb(&ctx->pn[i].v.hevc, data, bytes, s_packet, sizeof(s_packet));
data = s_packet;
}
else if (MOV_OBJECT_AAC == ctx->pn[i].object)
{
int n = mpeg4_aac_adts_save(&ctx->pn[i].a.aac, bytes, s_packet, sizeof(s_packet));
memcpy(s_packet + n, data, bytes);
data = s_packet;
bytes += n;
}
assert(0 == mpeg_ts_write(ctx->ts, ctx->pn[i].stream, (flags & MOV_AV_FLAG_KEYFREAME) ? 1 : 0, pts * 90, dts * 90, data, bytes));
return;
}
}
}
static void mov_video_info(void* param, uint32_t track, uint8_t object, int /*width*/, int /*height*/, const void* extra, size_t bytes)
{
assert(MOV_OBJECT_H264 == object || MOV_OBJECT_HEVC == object);
mpeg_ts_multi_program_test_t* ctx = (mpeg_ts_multi_program_test_t*)param;
ctx->pn[ctx->count].pn = ctx->count + 1;
ctx->pn[ctx->count].track = track;
ctx->pn[ctx->count].object = object;
assert(0 == mpeg_ts_add_program(ctx->ts, ctx->pn[ctx->count].pn, NULL, 0));
switch (object)
{
case MOV_OBJECT_H264:
assert(mpeg4_avc_decoder_configuration_record_load((const uint8_t*)extra, bytes, &ctx->pn[ctx->count].v.avc) > 0);
ctx->pn[ctx->count].stream = mpeg_ts_add_program_stream(ctx->ts, ctx->pn[ctx->count].pn, PSI_STREAM_H264, NULL, 0);
break;
case MOV_OBJECT_HEVC:
assert(mpeg4_hevc_decoder_configuration_record_load((const uint8_t*)extra, bytes, &ctx->pn[ctx->count].v.hevc) > 0);
ctx->pn[ctx->count].stream = mpeg_ts_add_program_stream(ctx->ts, ctx->pn[ctx->count].pn, PSI_STREAM_H265, NULL, 0);
break;
default:
assert(0);
}
ctx->count++;
}
static void mov_audio_info(void* param, uint32_t track, uint8_t object, int channel_count, int /*bit_per_sample*/, int sample_rate, const void* extra, size_t bytes)
{
assert(MOV_OBJECT_AAC == object || MOV_OBJECT_OPUS == object);
mpeg_ts_multi_program_test_t* ctx = (mpeg_ts_multi_program_test_t*)param;
ctx->pn[ctx->count].pn = ctx->count + 1;
ctx->pn[ctx->count].track = track;
ctx->pn[ctx->count].object = object;
assert(0 == mpeg_ts_add_program(ctx->ts, ctx->pn[ctx->count].pn, NULL, 0));
assert(mpeg4_aac_audio_specific_config_load((const uint8_t*)extra, bytes, &ctx->pn[ctx->count].a.aac) > 0);
ctx->pn[ctx->count].stream = mpeg_ts_add_program_stream(ctx->ts, ctx->pn[ctx->count].pn, MOV_OBJECT_AAC == object ? PSI_STREAM_AAC : PSI_STREAM_AUDIO_OPUS, NULL, 0);
ctx->count++;
}
void mpeg_ts_multi_program_test(const char* mp4)
{
struct mpeg_ts_multi_program_test_t ctx;
memset(&ctx, 0, sizeof(ctx));
char output[256] = { 0 };
snprintf(output, sizeof(output) - 1, "%s.ts", mp4);
struct mpeg_ts_func_t tshandler;
tshandler.alloc = ts_alloc;
tshandler.write = ts_write;
tshandler.free = ts_free;
FILE* fp = fopen(output, "wb");
ctx.ts = mpeg_ts_create(&tshandler, fp);
FILE* rfp = fopen(mp4, "rb");
mov_reader_t* mov = mov_reader_create(mov_file_buffer(), rfp);
struct mov_reader_trackinfo_t info = { mov_video_info, mov_audio_info };
mov_reader_getinfo(mov, &info, &ctx);
while (mov_reader_read(mov, s_buffer, sizeof(s_buffer), onread, &ctx) > 0)
{
}
mov_reader_destroy(mov);
mpeg_ts_destroy(ctx.ts);
fclose(rfp);
fclose(fp);
}