#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 #include #include #include 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); }