mirror of
https://github.com/mpv-player/mpv
synced 2025-01-04 14:12:10 +00:00
6aad532aa3
Remove them from the big MPOpts struct and move them to their sub structs. In the places where their fields are used, create a private copy of the structs, instead of accessing the semi-deprecated global option struct instance (mpv_global.opts) directly. This actually makes accessing these options finally thread-safe. They weren't even if they should have for years. (Including some potential for undefined behavior when e.g. the OSD font was changed at runtime.) This is mostly transparent. All options get moved around, but most users of the options just need to access a different struct (changing sd.opts to a different type changes a lot of uses, for example). One thing which has to be considered and could cause potential regressions is that the new option copies must be explicitly updated. sub_update_opts() takes care of this for example. Another thing is that writing to the option structs manually won't work, because the changes won't be propagated to other copies. Apparently the only affected case is the implementation of the sub-step command, which tries to change sub_delay. Handle this one explicitly (osd_changed() doesn't need to be called anymore, because changing the option triggers UPDATE_OSD, and updates the OSD as a consequence). The way the option value is propagated is rather hacky, but for now this will do.
362 lines
12 KiB
C
362 lines
12 KiB
C
/*
|
|
* This file is part of mpv.
|
|
*
|
|
* mpv 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.
|
|
*
|
|
* mpv 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 mpv. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include <stddef.h>
|
|
#include <stdbool.h>
|
|
#include <errno.h>
|
|
#include <assert.h>
|
|
|
|
#include "config.h"
|
|
#include "mpv_talloc.h"
|
|
|
|
#include "osdep/io.h"
|
|
#include "osdep/timer.h"
|
|
#include "osdep/threads.h"
|
|
|
|
#include "common/msg.h"
|
|
#include "options/options.h"
|
|
#include "options/m_property.h"
|
|
#include "options/m_config.h"
|
|
#include "common/common.h"
|
|
#include "common/global.h"
|
|
#include "common/encode.h"
|
|
#include "common/playlist.h"
|
|
#include "input/input.h"
|
|
|
|
#include "audio/out/ao.h"
|
|
#include "demux/demux.h"
|
|
#include "stream/stream.h"
|
|
#include "video/out/vo.h"
|
|
|
|
#include "core.h"
|
|
#include "command.h"
|
|
|
|
double rel_time_to_abs(struct MPContext *mpctx, struct m_rel_time t)
|
|
{
|
|
double length = get_time_length(mpctx);
|
|
// declaration up here because of C grammar quirk
|
|
double chapter_start_pts;
|
|
switch (t.type) {
|
|
case REL_TIME_ABSOLUTE:
|
|
return t.pos;
|
|
case REL_TIME_RELATIVE:
|
|
if (t.pos >= 0) {
|
|
return t.pos;
|
|
} else {
|
|
if (length >= 0)
|
|
return MPMAX(length + t.pos, 0.0);
|
|
}
|
|
break;
|
|
case REL_TIME_PERCENT:
|
|
if (length >= 0)
|
|
return length * (t.pos / 100.0);
|
|
break;
|
|
case REL_TIME_CHAPTER:
|
|
chapter_start_pts = chapter_start_time(mpctx, t.pos);
|
|
if (chapter_start_pts != MP_NOPTS_VALUE){
|
|
/*
|
|
* rel_time_to_abs always returns rebased timetamps,
|
|
* even with --rebase-start-time=no. (See the above two
|
|
* cases.) chapter_start_time values are not rebased without
|
|
* --rebase-start-time=yes, so we need to rebase them
|
|
* here to be consistent with the rest of rel_time_to_abs.
|
|
*/
|
|
if (mpctx->demuxer && !mpctx->opts->rebase_start_time){
|
|
chapter_start_pts -= mpctx->demuxer->start_time;
|
|
}
|
|
return chapter_start_pts;
|
|
}
|
|
break;
|
|
}
|
|
return MP_NOPTS_VALUE;
|
|
}
|
|
|
|
double get_play_end_pts(struct MPContext *mpctx)
|
|
{
|
|
struct MPOpts *opts = mpctx->opts;
|
|
double end = MP_NOPTS_VALUE;
|
|
if (opts->play_end.type) {
|
|
end = rel_time_to_abs(mpctx, opts->play_end);
|
|
}
|
|
if (opts->play_length.type) {
|
|
double start = get_play_start_pts(mpctx);
|
|
if (start == MP_NOPTS_VALUE)
|
|
start = 0;
|
|
double length = rel_time_to_abs(mpctx, opts->play_length);
|
|
if (length != MP_NOPTS_VALUE && (end == MP_NOPTS_VALUE || start + length < end))
|
|
end = start + length;
|
|
}
|
|
if (opts->chapterrange[1] > 0) {
|
|
double cend = chapter_start_time(mpctx, opts->chapterrange[1]);
|
|
if (cend != MP_NOPTS_VALUE && (end == MP_NOPTS_VALUE || cend < end))
|
|
end = cend;
|
|
}
|
|
// even though MP_NOPTS_VALUE is currently negative
|
|
// it doesn't necessarily have to remain that way
|
|
double ab_loop_start_time = get_ab_loop_start_time(mpctx);
|
|
if (mpctx->ab_loop_clip && opts->ab_loop[1] != MP_NOPTS_VALUE &&
|
|
(ab_loop_start_time == MP_NOPTS_VALUE || opts->ab_loop[1] > ab_loop_start_time))
|
|
{
|
|
if (end == MP_NOPTS_VALUE || end > opts->ab_loop[1])
|
|
end = opts->ab_loop[1];
|
|
}
|
|
return end;
|
|
}
|
|
|
|
/**
|
|
* Get the rebased PTS for which playback should start.
|
|
* The order of priority is as follows:
|
|
* 1. --start, if set.
|
|
* 2. The start chapter, if set.
|
|
* 3. MP_NOPTS_VALUE.
|
|
* If unspecified, return MP_NOPTS_VALUE.
|
|
* Does not return zero unless the start time is explicitly set to zero.
|
|
*/
|
|
double get_play_start_pts(struct MPContext *mpctx)
|
|
{
|
|
struct MPOpts *opts = mpctx->opts;
|
|
double play_start_pts = rel_time_to_abs(mpctx, opts->play_start);
|
|
if (play_start_pts == MP_NOPTS_VALUE && opts->chapterrange[0] > 0) {
|
|
double chapter_start_pts = chapter_start_time(mpctx, opts->chapterrange[0] - 1);
|
|
if (chapter_start_pts != MP_NOPTS_VALUE) {
|
|
/*
|
|
* get_play_start_pts always returns rebased timetamps,
|
|
* even with --rebase-start-time=no. chapter_start_time
|
|
* values are not rebased without --rebase-start-time=yes,
|
|
* so we need to rebase them here to be consistent with
|
|
* the rest of get_play_start_pts.
|
|
*/
|
|
if (mpctx->demuxer && !mpctx->opts->rebase_start_time){
|
|
chapter_start_pts -= mpctx->demuxer->start_time;
|
|
}
|
|
play_start_pts = chapter_start_pts;
|
|
}
|
|
}
|
|
return play_start_pts;
|
|
}
|
|
|
|
/**
|
|
* Get the time that an ab-loop seek should seek to.
|
|
* The order of priority is as follows:
|
|
* 1. --ab-loop-a, if set.
|
|
* 2. The Playback Start PTS, if set.
|
|
* 3. MP_NOPTS_VALUE.
|
|
* If unspecified, return MP_NOPTS_VALUE.
|
|
* Does not return zero unless the start time is explicitly set to zero.
|
|
*/
|
|
double get_ab_loop_start_time(struct MPContext *mpctx)
|
|
{
|
|
struct MPOpts *opts = mpctx->opts;
|
|
double ab_loop_start_time;
|
|
if (opts->ab_loop[0] != MP_NOPTS_VALUE) {
|
|
ab_loop_start_time = opts->ab_loop[0];
|
|
} else {
|
|
/*
|
|
* There is no check for MP_NOPTS_VALUE here
|
|
* because that's exactly what we want to return
|
|
* if get_play_start_pts comes up empty here.
|
|
*/
|
|
ab_loop_start_time = get_play_start_pts(mpctx);
|
|
}
|
|
return ab_loop_start_time;
|
|
}
|
|
|
|
double get_track_seek_offset(struct MPContext *mpctx, struct track *track)
|
|
{
|
|
struct MPOpts *opts = mpctx->opts;
|
|
if (track->selected) {
|
|
if (track->type == STREAM_AUDIO)
|
|
return -opts->audio_delay;
|
|
if (track->type == STREAM_SUB)
|
|
return -opts->subs_rend->sub_delay;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
void issue_refresh_seek(struct MPContext *mpctx, enum seek_precision min_prec)
|
|
{
|
|
// let queued seeks execute at a slightly later point
|
|
if (mpctx->seek.type) {
|
|
mp_wakeup_core(mpctx);
|
|
return;
|
|
}
|
|
// repeat currently ongoing seeks
|
|
if (mpctx->current_seek.type) {
|
|
mpctx->seek = mpctx->current_seek;
|
|
mp_wakeup_core(mpctx);
|
|
return;
|
|
}
|
|
// maybe happens when changing filters while file is loaded - ignore for now
|
|
if (mpctx->playback_pts == MP_NOPTS_VALUE)
|
|
return;
|
|
queue_seek(mpctx, MPSEEK_ABSOLUTE, mpctx->playback_pts, min_prec, 0);
|
|
}
|
|
|
|
float mp_get_cache_percent(struct MPContext *mpctx)
|
|
{
|
|
struct stream_cache_info info = {0};
|
|
if (mpctx->demuxer)
|
|
demux_stream_control(mpctx->demuxer, STREAM_CTRL_GET_CACHE_INFO, &info);
|
|
if (info.size > 0 && info.fill >= 0)
|
|
return info.fill / (info.size / 100.0);
|
|
return -1;
|
|
}
|
|
|
|
bool mp_get_cache_idle(struct MPContext *mpctx)
|
|
{
|
|
struct stream_cache_info info = {0};
|
|
if (mpctx->demuxer)
|
|
demux_stream_control(mpctx->demuxer, STREAM_CTRL_GET_CACHE_INFO, &info);
|
|
return info.idle;
|
|
}
|
|
|
|
void update_vo_playback_state(struct MPContext *mpctx)
|
|
{
|
|
if (mpctx->video_out && mpctx->video_out->config_ok) {
|
|
struct voctrl_playback_state oldstate = mpctx->vo_playback_state;
|
|
struct voctrl_playback_state newstate = {
|
|
.taskbar_progress = mpctx->opts->vo->taskbar_progress,
|
|
.playing = mpctx->playing,
|
|
.paused = mpctx->paused,
|
|
.percent_pos = get_percent_pos(mpctx),
|
|
};
|
|
|
|
if (oldstate.taskbar_progress != newstate.taskbar_progress ||
|
|
oldstate.playing != newstate.playing ||
|
|
oldstate.paused != newstate.paused ||
|
|
oldstate.percent_pos != newstate.percent_pos)
|
|
{
|
|
// Don't update progress bar if it was and still is hidden
|
|
if ((oldstate.playing && oldstate.taskbar_progress) ||
|
|
(newstate.playing && newstate.taskbar_progress))
|
|
{
|
|
vo_control_async(mpctx->video_out,
|
|
VOCTRL_UPDATE_PLAYBACK_STATE, &newstate);
|
|
}
|
|
mpctx->vo_playback_state = newstate;
|
|
}
|
|
} else {
|
|
mpctx->vo_playback_state = (struct voctrl_playback_state){ 0 };
|
|
}
|
|
}
|
|
|
|
void update_window_title(struct MPContext *mpctx, bool force)
|
|
{
|
|
if (!mpctx->video_out && !mpctx->ao) {
|
|
talloc_free(mpctx->last_window_title);
|
|
mpctx->last_window_title = NULL;
|
|
return;
|
|
}
|
|
char *title = mp_property_expand_string(mpctx, mpctx->opts->wintitle);
|
|
if (!mpctx->last_window_title || force ||
|
|
strcmp(title, mpctx->last_window_title) != 0)
|
|
{
|
|
talloc_free(mpctx->last_window_title);
|
|
mpctx->last_window_title = talloc_steal(mpctx, title);
|
|
|
|
if (mpctx->video_out)
|
|
vo_control(mpctx->video_out, VOCTRL_UPDATE_WINDOW_TITLE, title);
|
|
|
|
if (mpctx->ao) {
|
|
ao_control(mpctx->ao, AOCONTROL_UPDATE_STREAM_TITLE, title);
|
|
}
|
|
} else {
|
|
talloc_free(title);
|
|
}
|
|
}
|
|
|
|
void error_on_track(struct MPContext *mpctx, struct track *track)
|
|
{
|
|
if (!track || !track->selected)
|
|
return;
|
|
mp_deselect_track(mpctx, track);
|
|
if (track->type == STREAM_AUDIO)
|
|
MP_INFO(mpctx, "Audio: no audio\n");
|
|
if (track->type == STREAM_VIDEO)
|
|
MP_INFO(mpctx, "Video: no video\n");
|
|
if (mpctx->opts->stop_playback_on_init_failure ||
|
|
!(mpctx->vo_chain || mpctx->ao_chain))
|
|
{
|
|
if (!mpctx->stop_play)
|
|
mpctx->stop_play = PT_ERROR;
|
|
if (mpctx->error_playing >= 0)
|
|
mpctx->error_playing = MPV_ERROR_NOTHING_TO_PLAY;
|
|
}
|
|
mp_wakeup_core(mpctx);
|
|
}
|
|
|
|
int stream_dump(struct MPContext *mpctx, const char *source_filename)
|
|
{
|
|
struct MPOpts *opts = mpctx->opts;
|
|
stream_t *stream = stream_open(source_filename, mpctx->global);
|
|
if (!stream)
|
|
return -1;
|
|
|
|
int64_t size = stream_get_size(stream);
|
|
|
|
FILE *dest = fopen(opts->stream_dump, "wb");
|
|
if (!dest) {
|
|
MP_ERR(mpctx, "Error opening dump file: %s\n", mp_strerror(errno));
|
|
return -1;
|
|
}
|
|
|
|
bool ok = true;
|
|
|
|
while (mpctx->stop_play == KEEP_PLAYING && ok) {
|
|
if (!opts->quiet && ((stream->pos / (1024 * 1024)) % 2) == 1) {
|
|
uint64_t pos = stream->pos;
|
|
MP_MSG(mpctx, MSGL_STATUS, "Dumping %lld/%lld...",
|
|
(long long int)pos, (long long int)size);
|
|
}
|
|
bstr data = stream_peek(stream, STREAM_MAX_BUFFER_SIZE);
|
|
if (data.len == 0) {
|
|
ok &= stream->eof;
|
|
break;
|
|
}
|
|
ok &= fwrite(data.start, data.len, 1, dest) == 1;
|
|
stream_skip(stream, data.len);
|
|
mp_wakeup_core(mpctx); // don't actually sleep
|
|
mp_idle(mpctx); // but process input
|
|
}
|
|
|
|
ok &= fclose(dest) == 0;
|
|
free_stream(stream);
|
|
return ok ? 0 : -1;
|
|
}
|
|
|
|
void merge_playlist_files(struct playlist *pl)
|
|
{
|
|
if (!pl->first)
|
|
return;
|
|
char *edl = talloc_strdup(NULL, "edl://");
|
|
for (struct playlist_entry *e = pl->first; e; e = e->next) {
|
|
if (e != pl->first)
|
|
edl = talloc_strdup_append_buffer(edl, ";");
|
|
// Escape if needed
|
|
if (e->filename[strcspn(e->filename, "=%,;\n")] ||
|
|
bstr_strip(bstr0(e->filename)).len != strlen(e->filename))
|
|
{
|
|
// %length%
|
|
edl = talloc_asprintf_append_buffer(edl, "%%%zd%%", strlen(e->filename));
|
|
}
|
|
edl = talloc_strdup_append_buffer(edl, e->filename);
|
|
}
|
|
playlist_clear(pl);
|
|
playlist_add_file(pl, edl);
|
|
talloc_free(edl);
|
|
}
|