1
0
Fork 0
libbluray/src/devtools/mpls_dump.c

782 lines
23 KiB
C

/*
* This file is part of libbluray
* Copyright (C) 2009-2010 John Stebbins
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public
* License along with this library. If not, see
* <http://www.gnu.org/licenses/>.
*/
#include <sys/stat.h>
#include <dirent.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <libgen.h>
#include "bdnav/mpls_data.h"
#include "bluray.h"
#include "util.h"
#include "strings.h"
#ifdef _WIN32
# define DIR_SEP "\\"
# define PLAYLIST_DIR "\\BDMV\\PLAYLIST"
#else
# define DIR_SEP "/"
# define PLAYLIST_DIR "/BDMV/PLAYLIST"
#endif
static int verbose;
const VALUE_MAP subpath_type_map[] = {
{2, "Primary audio of the Browsable slideshow"},
{3, "Interactive Graphics presentation menu"},
{4, "Text Subtitle"},
{5, "Out-of-mux Synchronous elementary streams"},
{6, "Out-of-mux Asynchronous Picture-in-Picture presentation"},
{7, "In-mux Synchronous Picture-in-Picture presentation"},
{8, "SS Video"},
{0,NULL}
};
const VALUE_MAP playback_type_map[] = {
{1, "Sequential"},
{2, "Random"},
{3, "Shuffle"},
{0, NULL}
};
const VALUE_MAP connection_type_map[] = {
{1, "Non-seamless"},
{5, "Seamless"},
{6, "Seamless"},
{0, NULL}
};
const VALUE_MAP mark_type_map[] = {
{BLURAY_MARK_ENTRY, "Entry (chapter)"},
{BLURAY_MARK_LINK, "Link"},
{0, NULL}
};
static char *
_mk_path(const char *base, const char *sub)
{
size_t n1 = strlen(base);
size_t n2 = strlen(sub);
char *result = (char*)malloc(n1 + n2 + strlen(DIR_SEP) + 1);
if (result) {
strcpy(result, base);
strcat(result, DIR_SEP);
strcat(result, sub);
}
return result;
}
static void
_show_stream(MPLS_STREAM *ss, int level)
{
indent_printf(level, "Codec (%04x): %s", ss->coding_type,
_lookup_str(codec_map, ss->coding_type));
switch (ss->stream_type) {
case 1:
indent_printf(level, "PID: %04x", ss->pid);
break;
case 2:
case 4:
indent_printf(level, "SubPath Id: %02x", ss->subpath_id);
indent_printf(level, "SubClip Id: %02x", ss->subclip_id);
indent_printf(level, "PID: %04x", ss->pid);
break;
case 3:
indent_printf(level, "SubPath Id: %02x", ss->subpath_id);
indent_printf(level, "PID: %04x", ss->pid);
break;
default:
fprintf(stderr, "unrecognized stream type %02x\n", ss->stream_type);
break;
};
switch (ss->coding_type) {
case 0x01:
case 0x02:
case 0xea:
case 0x1b:
case 0x24:
indent_printf(level, "Format %02x: %s", ss->format,
_lookup_str(video_format_map, ss->format));
indent_printf(level, "Rate %02x: %s", ss->rate,
_lookup_str(video_rate_map, ss->rate));
break;
case 0x03:
case 0x04:
case 0x80:
case 0x81:
case 0x82:
case 0x83:
case 0x84:
case 0x85:
case 0x86:
case 0xa1:
case 0xa2:
indent_printf(level, "Format %02x: %s", ss->format,
_lookup_str(audio_format_map, ss->format));
indent_printf(level, "Rate %02x: %s", ss->rate,
_lookup_str(audio_rate_map, ss->rate));
indent_printf(level, "Language: %s", ss->lang);
break;
case 0x90:
case 0x91:
indent_printf(level, "Language: %s", ss->lang);
break;
case 0x92:
indent_printf(level, "Char Code: %02x", ss->char_code);
indent_printf(level, "Language: %s", ss->lang);
break;
default:
fprintf(stderr, "unrecognized coding type %02x\n", ss->coding_type);
break;
};
}
static void
_show_details(MPLS_PL *pl, int level)
{
int ii, jj, kk;
for (ii = 0; ii < pl->list_count; ii++) {
MPLS_PI *pi;
pi = &pl->play_item[ii];
indent_printf(level, "Clip Id %s", pi->clip[0].clip_id);
indent_printf(level+1, "Stc Id: %02x", pi->clip[0].stc_id);
indent_printf(level+1, "Connection Condition: %s (%02x)",
_lookup_str(connection_type_map, pi->connection_condition),
pi->connection_condition);
indent_printf(level+1, "In-Time: %d", pi->in_time);
indent_printf(level+1, "Out-Time: %d", pi->out_time);
if (pi->still_mode == 1) {
indent_printf(level+1, "Still time: %ds\n", pi->still_time);
}
if (pi->still_mode == 2) {
indent_printf(level+1, "Still time: infinite\n");
}
if (pi->angle_count > 1) {
for (jj = 1; jj < pi->angle_count; jj++) {
indent_printf(level+1, "Angle %d:", jj);
indent_printf(level+2, "Clip Id %s", pi->clip[jj].clip_id);
indent_printf(level+2, "Stc Id: %02x", pi->clip[jj].stc_id);
}
}
for (jj = 0; jj < pi->stn.num_video; jj++) {
indent_printf(level+1, "Video Stream %d:", jj);
_show_stream(&pi->stn.video[jj], level + 2);
}
for (jj = 0; jj < pi->stn.num_audio; jj++) {
indent_printf(level+1, "Audio Stream %d:", jj);
_show_stream(&pi->stn.audio[jj], level + 2);
}
for (jj = 0; jj < pi->stn.num_ig; jj++) {
indent_printf(level+1, "Interactive Graphics Stream %d:", jj);
_show_stream(&pi->stn.ig[jj], level + 2);
}
for (jj = 0; jj < (pi->stn.num_pg + pi->stn.num_pip_pg); jj++) {
if (jj < pi->stn.num_pg) {
indent_printf(level+1, "Presentation Graphics Stream %d:", jj);
} else {
indent_printf(level+1, "PIP Presentation Graphics Stream %d:", jj);
}
_show_stream(&pi->stn.pg[jj], level + 2);
}
for (jj = 0; jj < pi->stn.num_secondary_video; jj++) {
indent_printf(level+1, "Secondary Video Stream %d:", jj);
_show_stream(&pi->stn.secondary_video[jj], level + 2);
for (kk = 0; kk < pi->stn.secondary_video[jj].sv_num_secondary_audio_ref; kk++) {
indent_printf(level+2, "Secondary Audio Ref %d: %d", kk,pi->stn.secondary_video[jj].sv_secondary_audio_ref[kk]);
}
for (kk = 0; kk < pi->stn.secondary_video[jj].sv_num_pip_pg_ref; kk++) {
indent_printf(level+2, "PIP Presentation Graphic Ref %d: %d", kk,pi->stn.secondary_video[jj].sv_pip_pg_ref[kk]);
}
}
for (jj = 0; jj < pi->stn.num_secondary_audio; jj++) {
indent_printf(level+1, "Secondary Audio Stream %d:", jj);
_show_stream(&pi->stn.secondary_audio[jj], level + 2);
for (kk = 0; kk < pi->stn.secondary_audio[jj].sa_num_primary_audio_ref; kk++) {
indent_printf(level+2, "Primary Audio Ref %d: %d", kk,pi->stn.secondary_audio[jj].sa_primary_audio_ref[kk]);
}
}
printf("\n");
}
}
static void
_show_ai(MPLS_PL *pl, int level)
{
indent_printf(level, "Playback type: %s (%d)",
_lookup_str(playback_type_map, pl->app_info.playback_type),
pl->app_info.playback_type);
if (pl->app_info.playback_type == 2 || pl->app_info.playback_type == 3) {
indent_printf(level+1, "Playback count: %d", pl->app_info.playback_count);
}
}
static void
_show_marks(MPLS_PL *pl, int level)
{
int ii;
indent_printf(level, "PlayMark Count %d", pl->mark_count);
for (ii = 0; ii < pl->mark_count; ii++) {
MPLS_PI *pi;
MPLS_PLM *plm;
int min;
double sec;
plm = &pl->play_mark[ii];
indent_printf(level, "PlayMark %d", ii);
indent_printf(level+1, "Type: %s (%02x)",
_lookup_str(mark_type_map, plm->mark_type),
plm->mark_type);
if (plm->play_item_ref < pl->list_count) {
pi = &pl->play_item[plm->play_item_ref];
indent_printf(level+1, "PlayItem: %s", pi->clip[0].clip_id);
} else {
indent_printf(level+1, "PlayItem: Invalid reference");
}
indent_printf(level+1, "Time (ticks): %u", plm->time);
min = plm->duration / (45000*60);
sec = (double)(plm->duration - min * 45000 * 60) / 45000;
indent_printf(level+1, "Duration (mm:ss.ms, ticks): %d:%.2f, %u",
min, sec, plm->duration);
printf("\n");
}
}
static void
_show_clip_list(MPLS_PL *pl, int level)
{
int ii, jj;
for (ii = 0; ii < pl->list_count; ii++) {
MPLS_PI *pi;
pi = &pl->play_item[ii];
if (verbose) {
uint32_t duration;
duration = pi->out_time - pi->in_time;
indent_printf(level, "%s.m2ts -- Duration: %3d:%02d",
pi->clip[0].clip_id,
duration / (45000 * 60), (duration / 45000) % 60);
} else {
indent_printf(level, "%s.m2ts", pi->clip[0].clip_id);
}
if (pi->angle_count > 1) {
for (jj = 1; jj < pi->angle_count; jj++) {
indent_printf(level+1, "Angle %d: %s.m2ts", jj+1, pi->clip[jj].clip_id);
}
}
}
printf("\n");
}
static void
_show_sub_path(MPLS_SUB *sub, int level)
{
int ii;
indent_printf(level+1, "Type: %d (%s)", sub->type, _lookup_str(subpath_type_map, sub->type));
indent_printf(level+1, "Repeat: %d", sub->is_repeat);
indent_printf(level+1, "Sub playitem count: %d", sub->sub_playitem_count);
for (ii = 0; ii < sub->sub_playitem_count; ii++) {
MPLS_SUB_PI *pi;
pi = &sub->sub_play_item[ii];
if (verbose) {
indent_printf(level+1, "Sub playitem %d", ii);
indent_printf(level+2, "Clip Id %s", pi->clip[0].clip_id);
indent_printf(level+2, "Multi clip: %d", pi->is_multi_clip);
indent_printf(level+2, "Clip count: %d", pi->clip_count);
indent_printf(level+2, "Connection Condition: %s (%02x)",
_lookup_str(connection_type_map, pi->connection_condition),
pi->connection_condition);
indent_printf(level+2, "In-Time: %d", pi->in_time);
indent_printf(level+2, "Out-Time: %d", pi->out_time);
indent_printf(level+2, "Sync playitem Id: %d", pi->sync_play_item_id);
indent_printf(level+2, "Sync PTS: %d", pi->sync_pts);
} else {
indent_printf(level+1, "%s.m2ts", pi->clip[0].clip_id);
}
}
}
static void
_show_pip_metadata_block(MPLS_PIP_METADATA *block, int level)
{
int ii;
indent_printf(level, "Clip ref: %d", block->clip_ref);
indent_printf(level, "Secondary video ref: %d", block->secondary_video_ref);
indent_printf(level, "Timeline type: %d", block->timeline_type);
indent_printf(level, "Luma key flag: %d", block->luma_key_flag);
if (block->luma_key_flag) {
indent_printf(level, "Upper limit luma key: %d", block->upper_limit_luma_key);
}
indent_printf(level, "Trick play flag: %d", block->trick_play_flag);
for (ii = 0; ii < block->data_count; ii++) {
indent_printf(level, "data block %d:", ii);
indent_printf(level+1, "Timestamp: %d", block->data[ii].time);
indent_printf(level+1, "Horizontal position %d", block->data[ii].xpos);
indent_printf(level+1, "Vertical position: %d", block->data[ii].ypos);
indent_printf(level+1, "Scaling factor: %d", block->data[ii].scale_factor);
}
}
static void
_show_pip_metadata(MPLS_PL *pl, int level)
{
int ii;
for (ii = 0; ii < pl->ext_pip_data_count; ii++) {
MPLS_PIP_METADATA *data;
data = &pl->ext_pip_data[ii];
indent_printf(level, "PiP metadata block %d:", ii);
_show_pip_metadata_block(data, level+1);
}
}
static void
_show_static_metadata_entry(MPLS_STATIC_METADATA *entry, int level)
{
indent_printf(level, "Dynamic Range Type: %d", entry->dynamic_range_type);
indent_printf(level, "Mastering Display Primary R (X, Y): (%f, %f)", (float)entry->display_primaries_x[primary_red]/50000L, (float)entry->display_primaries_y[primary_red]/50000L);
indent_printf(level, "Mastering Display Primary G (X, Y): (%f, %f)", (float)entry->display_primaries_x[primary_green]/50000L, (float)entry->display_primaries_y[primary_green]/50000L);
indent_printf(level, "Mastering Display Primary B (X, Y): (%f, %f)", (float)entry->display_primaries_x[primary_blue]/50000L, (float)entry->display_primaries_y[primary_blue]/50000L);
indent_printf(level, "White Point (X, Y): (%f, %f)", (float)entry->white_point_x/50000L, (float)entry->white_point_y/50000L);
indent_printf(level, "Display Mastering Luminance (min, max): (%.4f, %.4f)", (float)entry->min_display_mastering_luminance/10000L, (float)entry->max_display_mastering_luminance);
indent_printf(level, "Maximum Frame Average Light Level (MaxFALL): %d", entry->max_CLL);
indent_printf(level, "Maximum Content Light Level (MaxCLL): %d", entry->max_FALL);
}
static void
_show_static_metadata(MPLS_PL *pl, int level)
{
int ii;
for (ii = 0; ii < pl->ext_static_metadata_count; ii++) {
MPLS_STATIC_METADATA *data;
data = &pl->ext_static_metadata[ii];
indent_printf(level, "Static metadata entry %d:", ii);
_show_static_metadata_entry(data, level+1);
}
printf("\n");
}
static void
_show_sub_paths(MPLS_PL *pl, int level)
{
int ss;
for (ss = 0; ss < pl->sub_count; ss++) {
MPLS_SUB *sub;
sub = &pl->sub_path[ss];
indent_printf(level, "Sub Path %d:", ss);
_show_sub_path(sub, level+1);
}
}
static void
_show_sub_paths_ss(MPLS_PL *pl, int level)
{
int ss;
for (ss = 0; ss < pl->ext_sub_count; ss++) {
MPLS_SUB *sub;
sub = &pl->ext_sub_path[ss];
indent_printf(level, "Extension Sub Path %d:", ss);
_show_sub_path(sub, level+1);
}
printf("\n");
}
static uint32_t
_pl_duration(MPLS_PL *pl)
{
int ii;
uint32_t duration = 0;
MPLS_PI *pi;
for (ii = 0; ii < pl->list_count; ii++) {
pi = &pl->play_item[ii];
duration += pi->out_time - pi->in_time;
}
return duration;
}
static int
_filter_dup(MPLS_PL *pl_list[], int count, MPLS_PL *pl)
{
int ii, jj;
for (ii = 0; ii < count; ii++) {
if (pl->list_count != pl_list[ii]->list_count ||
_pl_duration(pl) != _pl_duration(pl_list[ii])) {
continue;
}
for (jj = 0; jj < pl->list_count; jj++) {
MPLS_PI *pi1, *pi2;
pi1 = &pl->play_item[jj];
pi2 = &pl_list[ii]->play_item[jj];
if (memcmp(pi1->clip[0].clip_id, pi2->clip[0].clip_id, 5) != 0 ||
pi1->in_time != pi2->in_time ||
pi1->out_time != pi2->out_time) {
break;
}
}
if (jj != pl->list_count) {
continue;
}
return 0;
}
return 1;
}
static int
_find_repeats(MPLS_PL *pl, const char *m2ts)
{
int ii, count = 0;
for (ii = 0; ii < pl->list_count; ii++) {
MPLS_PI *pi;
pi = &pl->play_item[ii];
// Ignore titles with repeated segments
if (strcmp(pi->clip[0].clip_id, m2ts) == 0) {
count++;
}
}
return count;
}
static int
_filter_short(MPLS_PL *pl, unsigned int seconds)
{
// Ignore short playlists
if (_pl_duration(pl) / 45000 <= seconds) {
return 0;
}
return 1;
}
static int
_filter_repeats(MPLS_PL *pl, int repeats)
{
int ii;
for (ii = 0; ii < pl->list_count; ii++) {
MPLS_PI *pi;
pi = &pl->play_item[ii];
// Ignore titles with repeated segments
if (_find_repeats(pl, pi->clip[0].clip_id) > repeats) {
return 0;
}
}
return 1;
}
static int clip_list = 0, playlist_info = 0, chapter_marks = 0, sub_paths = 0, pip_metadata = 0, static_metadata = 0;
static int repeats = 0, seconds = 0, dups = 0;
static MPLS_PL*
_process_file(char *name, MPLS_PL *pl_list[], int pl_count)
{
MPLS_PL *pl;
pl = bd_read_mpls(name);
if (pl == NULL) {
fprintf(stderr, "Parse failed: %s\n", name);
return NULL;
}
if (seconds) {
if (!_filter_short(pl, seconds)) {
bd_free_mpls(pl);
return NULL;
}
}
if (repeats) {
if (!_filter_repeats(pl, repeats)) {
bd_free_mpls(pl);
return NULL;
}
}
if (dups) {
if (!_filter_dup(pl_list, pl_count, pl)) {
bd_free_mpls(pl);
return NULL;
}
}
if (verbose) {
indent_printf(0,
"%s -- Num Clips: %3d , Duration: minutes %4u:%02u",
basename(name),
pl->list_count,
_pl_duration(pl) / (45000 * 60),
(_pl_duration(pl) / 45000) % 60);
_show_ai(pl, 1);
} else {
indent_printf(0, "%s -- Duration: minutes %4u:%02u",
basename(name),
_pl_duration(pl) / (45000 * 60),
(_pl_duration(pl) / 45000) % 60);
}
if (playlist_info) {
_show_details(pl, 1);
}
if (chapter_marks) {
_show_marks(pl, 1);
}
if (pip_metadata) {
_show_pip_metadata(pl, 1);
}
if (clip_list) {
_show_clip_list(pl, 1);
}
if (sub_paths) {
_show_sub_paths(pl, 1);
_show_sub_paths_ss(pl, 1);
}
if (static_metadata) {
_show_static_metadata(pl, 1);
}
return pl;
}
static void
_usage(char *cmd)
{
fprintf(stderr,
"Usage: %s -vli <mpls file> [<mpls file> ...]\n"
"With no options, produces a list of the playlist(s) with durations\n"
"Options:\n"
" v - Verbose output.\n"
" l - Produces a list of the m2ts clips\n"
" i - Dumps detailed information about each clip\n"
" c - Show chapter marks\n"
" p - Show sub paths\n"
" P - Show picture-in-picture metadata\n"
" S - Show static metadata\n"
" r <N> - Filter out titles that have >N repeating clips\n"
" d - Filter out duplicate titles\n"
" s <seconds> - Filter out short titles\n"
" f - Filter combination -r2 -d -s900\n"
, cmd);
exit(EXIT_FAILURE);
}
#define OPTS "vlicpPSfr:ds:"
static int
_qsort_str_cmp(const void *a, const void *b)
{
const char *stra = *(char * const *)a;
const char *strb = *(char * const *)b;
return strcmp(stra, strb);
}
int
main(int argc, char *argv[])
{
MPLS_PL *pl;
int opt;
int ii, pl_ii;
MPLS_PL *pl_list[1000];
struct stat st;
char *path = NULL;
DIR *dir = NULL;
do {
opt = getopt(argc, argv, OPTS);
switch (opt) {
case -1:
break;
case 'v':
verbose = 1;
break;
case 'l':
clip_list = 1;
break;
case 'i':
playlist_info = 1;
break;
case 'c':
chapter_marks = 1;
break;
case 'p':
sub_paths = 1;
break;
case 'P':
pip_metadata = 1;
break;
case 'S':
static_metadata = 1;
break;
case 'd':
dups = 1;
break;
case 'r':
repeats = atoi(optarg);
break;
case 'f':
repeats = 2;
dups = 1;
seconds = 900;
break;
case 's':
seconds = atoi(optarg);
break;
default:
_usage(argv[0]);
break;
}
} while (opt != -1);
if (optind >= argc) {
_usage(argv[0]);
}
for (pl_ii = 0, ii = optind; pl_ii < 1000 && ii < argc; ii++) {
if (stat(argv[ii], &st)) {
continue;
}
dir = NULL;
if (S_ISDIR(st.st_mode)) {
printf("Directory: %s:\n", argv[ii]);
/* drop old ones (do not check for duplicates across directories) */
for (int jj = 0; jj < pl_ii; jj++) {
bd_free_mpls(pl_list[jj]);
}
pl_ii = 0;
path = _mk_path(argv[ii], PLAYLIST_DIR);
if (path == NULL) {
fprintf(stderr, "Failed to find playlist path: %s\n", argv[ii]);
continue;
}
dir = opendir(path);
if (dir == NULL) {
fprintf(stderr, "Failed to open dir: %s\n", path);
free(path);
continue;
}
}
if (dir != NULL) {
char **dirlist = (char**)calloc(10001, sizeof(char*));
if (!dirlist) {
continue;
}
struct dirent *ent;
int jj = 0;
for (ent = readdir(dir); ent != NULL && jj < 1000; ent = readdir(dir)) {
char *s = (char*)malloc(strlen(ent->d_name) + 1);
if (s) {
dirlist[jj++] = strcpy(s, ent->d_name);
}
}
qsort(dirlist, jj, sizeof(char*), _qsort_str_cmp);
for (jj = 0; dirlist[jj] != NULL && pl_ii < 1000; jj++) {
char *name = NULL;
name = _mk_path(path, dirlist[jj]);
if (name == NULL) {
continue;
}
free(dirlist[jj]);
if (stat(name, &st)) {
free(name);
continue;
}
if (!S_ISREG(st.st_mode)) {
free(name);
continue;
}
pl = _process_file(name, pl_list, pl_ii);
free(name);
if (pl != NULL) {
pl_list[pl_ii++] = pl;
}
}
free(dirlist);
free(path);
closedir(dir);
dir = NULL;
} else {
pl = _process_file(argv[ii], pl_list, pl_ii);
if (pl != NULL) {
pl_list[pl_ii++] = pl;
}
}
if (pl_ii >= 999) {
fprintf(stderr, "Error: too many play lists given. Output is truncated.\n");
}
}
// Cleanup
for (ii = 0; ii < pl_ii; ii++) {
bd_free_mpls(pl_list[ii]);
}
return 0;
}