#include <unistd.h>
#include <fcntl.h>
#include <errno.h>
+#if !defined(__FreeBSD__)
#include <sys/stat.h>
#include <sys/statfs.h>
+#else
+#include <sys/param.h>
+#include <sys/mount.h>
+#endif
// BD Creation
{ "720x480 29.97p*", 720,480, 29.97, 0, ILACE_MODE_NOTINTERLACED },
};
+static struct bd_profile {
+ const char *name;
+} bd_profiles[] = {
+ {"bluray.m2ts"},
+ {"bluray_lpcm.m2ts"},
+};
+
+
const int64_t CreateBD_Thread::BD_SIZE = 25000000000;
const int CreateBD_Thread::BD_STREAMS = 1;
const int CreateBD_Thread::BD_WIDTH = 1920;
return ret;
}
+
+
char *BD_BatchRenderJob::create_script(EDL *edl, ArrayList<Indexable *> *idxbls)
{
char script[BCTEXTLEN];
strcpy(script, edl_path);
+
FILE *fp = 0;
char *bp = strrchr(script,'/');
int fd = -1;
fprintf(fp,"sz=`du -cb $dir/bd.m2ts* | tail -1 | sed -e 's/[ ].*//'`\n");
fprintf(fp,"blks=$((sz/2048 + 4096))\n");
fprintf(fp,"rm -f %s\n", udfs);
+ fprintf(fp,"if [ -f bd.meta ]; then\n");
+ fprintf(fp,"tsmuxer bd.meta $dir/bd.iso \n");
+ fprintf(fp,"mv $dir/bd.iso $dir/bd.udfs\n");
+ fprintf(fp,"else\n");
fprintf(fp,"mkudffs -b 2048 %s $blks\n", udfs);
fprintf(fp,"mount %s%s\n", mopts, mntpt);
fprintf(fp,"bdwrite %s $dir/bd.m2ts*\n",mntpt);
fprintf(fp,"umount %s\n",mntpt);
+ fprintf(fp,"fi\n");
if( is_usr_mnt )
fprintf(fp,"mv -f %s $dir/bd.udfs\n", udfs);
fprintf(fp,"echo To burn bluray, load writable media and run:\n");
this->use_resize_tracks = 0;
this->use_labeled = 0;
this->use_farmed = 0;
+ this->use_tsmuxer = 0;
+
+ strcpy(use_profile,"bluray.m2ts");
this->bd_size = BD_SIZE;
this->bd_width = BD_WIDTH;
this->bd_max_bitrate = BD_MAX_BITRATE;
this->bd_kaudio_rate = BD_KAUDIO_RATE;
this->max_w = this->max_h = 0;
+ this->batchrender = 0;
}
CreateBD_Thread::~CreateBD_Thread()
return 1;
}
+ if (use_tsmuxer) {
+ char meta_script[BCTEXTLEN];
+ strcpy(meta_script, "/");
+
+ FILE *fp = 0;
+ char *bp = strrchr(meta_script,'/');
+ int fd = -1;
+ if( bp ) {
+ char script_filename[BCTEXTLEN];
+ sprintf(script_filename, "%s/bd.meta", asset_dir);
+ strcpy(bp, script_filename);
+ fd = open(meta_script, O_WRONLY+O_CREAT+O_TRUNC, 0755);
+ }
+ if( fd >= 0 )
+ fp = fdopen(fd, "w");
+ if( !fp ) {
+ char err[BCTEXTLEN], msg[BCTEXTLEN];
+ strerror_r(errno, err, sizeof(err));
+ sprintf(msg, _("Unable to save: %s\n-- %s"), meta_script, err);
+ MainError::show_error(msg);
+ return 0;
+ }
+
+
+ fprintf(fp,"MUXOPT --blu-ray --hdmv-descriptors\n");
+ fprintf(fp,"V_MPEG4/ISO/AVC, bd.m2ts, track=4113\n");
+ if(!strcmp(use_profile, "bluray.m2ts"))
+ fprintf(fp,"A_AC3, bd.m2ts, track=4352\n");
+ if(!strcmp(use_profile, "bluray_lpcm.m2ts"))
+ fprintf(fp,"A_LPCM, bd.m2ts, track=4352\n");
+ fprintf(fp,"\n");
+ fclose(fp);
+
+ }
+
BatchRenderJob *job = new BD_BatchRenderJob(mwindow->preferences,
use_labeled, use_farmed);
jobs->append(job);
strcpy(asset->fformat, "m2ts");
asset->audio_data = 1;
- strcpy(asset->acodec, "bluray.m2ts");
+ strcpy(asset->acodec, use_profile);
//mwindow->defaults->get("DEFAULT_BLURAY_ACODEC", asset->acodec);
FFMPEG::set_option_path(option_path, "audio/%s", asset->acodec);
FFMPEG::load_options(option_path, asset->ff_audio_options,
asset->ff_audio_bitrate = bd_kaudio_rate * 1000;
asset->video_data = 1;
- const char *vcodec = "bluray.m2ts";
+ const char *vcodec = use_profile;
switch( asset->interlace_mode ) {
case ILACE_MODE_TOP_FIRST: vcodec = "bluray_tff.m2ts"; break;
case ILACE_MODE_BOTTOM_FIRST: vcodec = "bluray_bff.m2ts"; break;
char asset_dir[BCTEXTLEN], jobs_path[BCTEXTLEN];
snprintf(asset_dir, sizeof(asset_dir), "%s/%s", tmp_path, asset_title);
snprintf(jobs_path, sizeof(jobs_path), "%s/bd.jobs", asset_dir);
+ //batchrender->tsmuxered = use_tsmuxer;
mwindow->batch_render->reset(jobs_path);
int ret = create_bd_jobs(&mwindow->batch_render->jobs, asset_dir);
mwindow->undo->update_undo_after(_("create bd"), LOAD_ALL);
use_resize_tracks = 0;
use_labeled = 0;
use_farmed = 0;
+ use_tsmuxer = 0;
use_standard = !strcmp(mwindow->default_std(),"NTSC") ?
BD_1920x1080_2997i : BD_1920x1080_25i;
bd_size = BD_SIZE;
int scr_x = mwindow->gui->get_screen_x(0, -1);
int scr_w = mwindow->gui->get_screen_w(0, -1);
int scr_h = mwindow->gui->get_screen_h(0, -1);
- int w = xS(560), h = yS(290);
+ int w = xS(560), h = yS(340);
int x = scr_x + scr_w/2 - w/2, y = scr_h/2 - h/2;
gui = new CreateBD_GUI(this, x, y, w, h);
{
}
+CreateBD_UseTsmuxer::CreateBD_UseTsmuxer(CreateBD_GUI *gui, int x, int y)
+ : BC_CheckBox(x, y, &gui->thread->use_tsmuxer, _("use tsmuxer"))
+{
+ this->gui = gui;
+}
+
+CreateBD_UseTsmuxer::~CreateBD_UseTsmuxer()
+{
+}
+
CreateBD_GUI::CreateBD_GUI(CreateBD_Thread *thread, int x, int y, int w, int h)
: BC_Window(_(PROGRAM_NAME ": Create BD"), x, y, w, h, xS(50), yS(50), 1, 0, 1)
disk_space = 0;
standard = 0;
scale = 0;
+ profile = 0;
need_deinterlace = 0;
need_inverse_telecine = 0;
need_resize_tracks = 0;
need_wide_audio = 0;
need_labeled = 0;
need_farmed = 0;
+ need_tsmuxer = 0;
ok = 0;
cancel = 0;
// *** CONTEXT_HELP ***
void CreateBD_GUI::create_objects()
{
+ int xs1 = xS(1);
int xs10 = xS(10), xs35 = xS(35);
- int xs160 = xS(160), xs170 = xS(170);
+ int xs60 = xS(60), xs160 = xS(160), xs170 = xS(170);
int ys5 = yS(5), ys10 = yS(10);
lock_window("CreateBD_GUI::create_objects");
int pady = BC_TextBox::calculate_h(this, MEDIUMFONT, 0, 1) + ys5;
media_size->update(media_sizes[0]->get_text());
disk_space->update();
y += disk_space->get_h() + pady/2;
- title = new BC_Title(x, y, _("Format:"), MEDIUMFONT, YELLOW);
+
+ title = new BC_Title(x, y, _("Profile:"), MEDIUMFONT, YELLOW);
+ add_subwindow(title);
+ int start_x = x;
+ x += title->get_w()+padx;
+ profile = new CreateBD_Profile(this, x, y);
+ profile->create_objects();
+ profiles.append(new BC_ListBoxItem("bluray.m2ts"));
+ profiles.append(new BC_ListBoxItem("bluray_lpcm.m2ts"));
+/* profiles.append(new BC_ListBoxItem("bluray_truehd.m2ts")); */
+ profile->update_list(&profiles);
+ profile->update(profiles[0]->get_text());
+
+ x += profile->get_w()+padx;
+ need_tsmuxer = new CreateBD_UseTsmuxer(this, x, y);
+ add_subwindow(need_tsmuxer);
+ y += need_tsmuxer->get_h() + pady;
+
+ title = new BC_Title(start_x, y, _("Format:"), MEDIUMFONT, YELLOW);
add_subwindow(title);
standard = new CreateBD_Format(this, title->get_w() + padx, y);
add_subwindow(standard);
scale->create_objects();
y += standard->get_h() + pady/2;
x1 = x; int y1 = y;
- need_deinterlace = new CreateBD_Deinterlace(this, x1, y);
+ need_deinterlace = new CreateBD_Deinterlace(this, start_x, y);
add_subwindow(need_deinterlace);
y += need_deinterlace->get_h() + pady/2;
- need_histogram = new CreateBD_Histogram(this, x1, y);
+ need_histogram = new CreateBD_Histogram(this, start_x, y);
add_subwindow(need_histogram);
y += need_histogram->get_h() + pady/2;
non_standard = new BC_Title(x1, y+ys5, "", MEDIUMFONT, RED);
add_subwindow(non_standard);
- x1 += xs160; y = y1;
+ x1 -= xs60; y = y1;
need_inverse_telecine = new CreateBD_InverseTelecine(this, x1, y);
add_subwindow(need_inverse_telecine);
y += need_inverse_telecine->get_h() + pady/2;
y += need_wide_audio->get_h() + pady/2;
need_resize_tracks = new CreateBD_ResizeTracks(this, x1, y);
add_subwindow(need_resize_tracks);
- x1 += xs160; y = y1;
+ x1 += xs170; y = y1;
need_labeled = new CreateBD_LabelChapters(this, x1, y);
add_subwindow(need_labeled);
y += need_labeled->get_h() + pady/2;
add_subwindow(need_farmed);
ok_w = BC_OKButton::calculate_w();
ok_h = BC_OKButton::calculate_h();
- ok_x = xs10;
+ ok_x = xs1;
ok_y = get_h() - ok_h - xs10;
ok = new CreateBD_OK(this, ok_x, ok_y);
add_subwindow(ok);
return 1;
}
+CreateBD_Profile::CreateBD_Profile(CreateBD_GUI *gui, int x, int y)
+ : BC_PopupTextBox(gui, 0, 0, x, y, xS(170),yS(50))
+{
+ this->gui = gui;
+}
+
+CreateBD_Profile::~CreateBD_Profile()
+{
+}
+
+int CreateBD_Profile::handle_event()
+{
+ strcpy(gui->thread->use_profile, get_text());
+ return 1;
+}
#include <stdint.h>
#include <stdlib.h>
#include <string.h>
+#if !defined (__FreeBSD__)
#include <endian.h>
+#else
+#include <sys/endian.h>
+#endif
#include <limits.h>
#include <sys/stat.h>
// work arounds (centos)
BLURAY_STREAM_TYPE_VIDEO_VC1 = 0xea,
BLURAY_STREAM_TYPE_VIDEO_H264 = 0x1b,
BLURAY_STREAM_TYPE_VIDEO_H264_MVC = 0x20,
+ BLURAY_STREAM_TYPE_VIDEO_HEVC = 0x24,
BLURAY_STREAM_TYPE_SUB_PG = 0x90,
BLURAY_STREAM_TYPE_SUB_IG = 0x91,
BLURAY_STREAM_TYPE_SUB_TEXT = 0x92,
BLURAY_VIDEO_FORMAT_720P = 5, // SMPTE 296M
BLURAY_VIDEO_FORMAT_1080P = 6, // SMPTE 274M
BLURAY_VIDEO_FORMAT_576P = 7, // ITU-R BT.1358
+ BLURAY_VIDEO_FORMAT_2160P = 8,
BLURAY_VIDEO_RATE_24000_1001 = 1, // 23.976
BLURAY_VIDEO_RATE_24 = 2,
uint8_t subclip_id;
uint8_t format;
uint8_t rate;
+ uint8_t dynamic_range_type;
+ uint8_t color_space;
+ uint8_t cr_flag;
+ uint8_t hdr_plus_flag;
uint8_t char_code;
char lang[4];
ArrayList<mpls_pl *> pl;
void add_movie(uint32_t *ops, int n);
- int compose();
+ int compose(int ch_interval);
int write(char *fn);
Media() { path = 0; filename[0] = 0; }
case BLURAY_STREAM_TYPE_VIDEO_MPEG2:
case BLURAY_STREAM_TYPE_VIDEO_VC1:
case BLURAY_STREAM_TYPE_VIDEO_H264:
+ case BLURAY_STREAM_TYPE_VIDEO_HEVC:
case 0x20:
bs.write(format, 4);
bs.write(rate, 4);
bs.write(stream_type, 8);
switch (stream_type) {
- case 0x01:
+ case 1:
bs.write(pid, 16);
break;
- case 0x02:
- case 0x04:
+ case 2:
+ case 4:
bs.write(subpath_id, 8);
bs.write(subclip_id, 8);
bs.write(pid, 16);
break;
- case 0x03:
+ case 3:
bs.write(subpath_id, 8);
bs.write(pid, 16);
break;
default:
- fprintf(stderr, "unrecognized stream type %02x\n", stream_type);
+ fprintf(stderr, "unrecognized mpls stream type %02x\n", stream_type);
break;
};
bs.padb(9 - strm.bs_posb(bs));
case BLURAY_STREAM_TYPE_VIDEO_MPEG2:
case BLURAY_STREAM_TYPE_VIDEO_VC1:
case BLURAY_STREAM_TYPE_VIDEO_H264:
+ case BLURAY_STREAM_TYPE_VIDEO_HEVC:
bs.write(format, 4);
bs.write(rate, 4);
break;
const AVRational media_info::clk45k = { 1, 45000 };
-static int bd_stream_type(AVCodecID codec_id)
+static int bd_coding_type(AVCodecID codec_id)
{
- int stream_type = 0;
+ int coding_type = 0;
switch (codec_id) {
case AV_CODEC_ID_MPEG1VIDEO:
- stream_type = BLURAY_STREAM_TYPE_VIDEO_MPEG1;
+ coding_type = BLURAY_STREAM_TYPE_VIDEO_MPEG1;
break;
case AV_CODEC_ID_MPEG2VIDEO:
- stream_type = BLURAY_STREAM_TYPE_VIDEO_MPEG2;
+ coding_type = BLURAY_STREAM_TYPE_VIDEO_MPEG2;
break;
case AV_CODEC_ID_H264:
- stream_type = BLURAY_STREAM_TYPE_VIDEO_H264;
+ coding_type = BLURAY_STREAM_TYPE_VIDEO_H264;
+ break;
+ case AV_CODEC_ID_HEVC:
+ coding_type = BLURAY_STREAM_TYPE_VIDEO_HEVC;
break;
case AV_CODEC_ID_MP2:
- stream_type = BLURAY_STREAM_TYPE_AUDIO_MPEG1;
+ coding_type = BLURAY_STREAM_TYPE_AUDIO_MPEG1;
break;
case AV_CODEC_ID_MP3:
- stream_type = BLURAY_STREAM_TYPE_AUDIO_MPEG2;
+ coding_type = BLURAY_STREAM_TYPE_AUDIO_MPEG2;
break;
case AV_CODEC_ID_AC3:
- stream_type = BLURAY_STREAM_TYPE_AUDIO_AC3;
+ coding_type = BLURAY_STREAM_TYPE_AUDIO_AC3;
break;
case AV_CODEC_ID_EAC3:
- stream_type = BLURAY_STREAM_TYPE_AUDIO_AC3PLUS;
+ coding_type = BLURAY_STREAM_TYPE_AUDIO_AC3PLUS;
break;
case AV_CODEC_ID_DTS:
- stream_type = BLURAY_STREAM_TYPE_AUDIO_DTS;
+ coding_type = BLURAY_STREAM_TYPE_AUDIO_DTS;
break;
case AV_CODEC_ID_TRUEHD:
- stream_type = BLURAY_STREAM_TYPE_AUDIO_TRUHD;
+ coding_type = BLURAY_STREAM_TYPE_AUDIO_TRUHD;
+ break;
+ case AV_CODEC_ID_PCM_BLURAY:
+ coding_type = BLURAY_STREAM_TYPE_AUDIO_LPCM;
break;
case AV_CODEC_ID_HDMV_PGS_SUBTITLE:
- stream_type = BLURAY_STREAM_TYPE_SUB_PG;
+ coding_type = BLURAY_STREAM_TYPE_SUB_PG;
break;
default:
- fprintf(stderr, "unknown bluray stream type %s\n", avcodec_get_name(codec_id));
+ fprintf(stderr, "unknown bluray codec type %s\n", avcodec_get_name(codec_id));
exit(1);
}
- return stream_type;
+ return coding_type;
}
static int bd_audio_format(int channels)
if( w == 1280 && h == 720 /* && !ilace*/ ) return BLURAY_VIDEO_FORMAT_720P;
if( w == 1440 && h == 1080 /* && ilace*/ ) return BLURAY_VIDEO_FORMAT_1080I;
if( w == 1920 && h == 1080 /* && !ilace*/ ) return BLURAY_VIDEO_FORMAT_1080P;
+ if( w == 3840 && h == 2160 && !ilace ) return BLURAY_VIDEO_FORMAT_2160P;
+
fprintf(stderr, "unknown bluray video format %dx%d %silace\n",
w, h, !ilace ? "not " : "");
exit(1);
switch( type ) {
case AVMEDIA_TYPE_VIDEO: {
if( ep_pid < 0 ) ep_pid = st->id;
- s->coding_type = bd_stream_type(codec_id);
+ s->coding_type = bd_coding_type(codec_id);
int ilace = field_probe(fmt_ctx, st);
if( ilace < 0 ) {
fprintf(stderr, "interlace probe failed\n");
(double)st->sample_aspect_ratio.num / st->sample_aspect_ratio.den);
break; }
case AVMEDIA_TYPE_AUDIO: {
- s->coding_type = bd_stream_type(codec_id);
+ s->coding_type = bd_coding_type(codec_id);
s->format = bd_audio_format(st->codecpar->channels);
s->rate = bd_audio_rate(st->codecpar->sample_rate);
strcpy((char*)s->lang, "eng");
break; }
case AVMEDIA_TYPE_SUBTITLE: {
- s->coding_type = bd_stream_type(codec_id);
+ s->coding_type = bd_coding_type(codec_id);
AVDictionaryEntry *lang = av_dict_get(st->metadata, "language", 0, 0);
strncpy((char*)s->lang, lang ? lang->value : "und", sizeof(s->lang));
break; }
}
int
-Media::compose()
+Media::compose(int ch_interval)
{
// movie
bs.init();
}
pp->play_item.append(pi);
}
-// chapter marks every ch_duration ticks
- int64_t ch_duration = 45000 * 60*5;
+// chapter marks every ch_duration seconds * 45Kticks, default 5 min
+ int PCR_FREQ = 45000;
+ if (ch_interval == 0)
+ ch_interval = 60*5;
+ int64_t ch_duration = PCR_FREQ * ch_interval;
int64_t mrktm = ch_duration;
int64_t plytm = 0;
int pmark = 0, pitem = 0;
//av_log_set_level(AV_LOG_DEBUG);
Media media;
media_info *mp = 0;
+ int start = 0, chapter_every_n_sec = 0;
+
+ int opt = getopt(ac, av, "c:");
+ if (opt == 'c') {
+ chapter_every_n_sec = optarg[0]; start = 3; }
+ else
+ start = 2;
- for( int ii=2; ii<ac; ++ii ) {
+ for( int ii=start; ii<ac; ++ii ) {
char *ap = av[ii];
// any dash seq followed by number sets curr title pgm_pid
// single dash only sets title pgm_pid
if( mp ) mp->brk = 1;
- if( media.compose() ) {
+ if( media.compose(chapter_every_n_sec) ) {
fprintf(stderr, "cant compose media\n");
return 1;
}