Update the video matrix tagging scheme

Write the color range and color space used to convert video to RGB to
the YCbCr Matrix field, or None if no non-RGB video was opened. See
http://code.google.com/p/xy-vsfilter/issues/detail?id=91 for related
discussion.

Originally committed to SVN as r6762.
This commit is contained in:
Thomas Goyne 2012-05-11 14:38:44 +00:00
parent baacea6ab0
commit bc74dfc33b
8 changed files with 41 additions and 20 deletions

View file

@ -29,6 +29,7 @@ function clean_script_info(subs)
["scaledborderandshadow"] = true, ["scaledborderandshadow"] = true,
["scripttype"] = true, ["scripttype"] = true,
["video colorspace"] = true, ["video colorspace"] = true,
["ycbcr matrix"] = true,
["wrapstyle"] = true ["wrapstyle"] = true
} }

View file

@ -378,6 +378,7 @@ void AssFile::LoadDefault(bool defline) {
AddLine(wxString::Format("PlayResX: %" PRId64, OPT_GET("Subtitle/Default Resolution/Width")->GetInt()), &version, &attach); AddLine(wxString::Format("PlayResX: %" PRId64, OPT_GET("Subtitle/Default Resolution/Width")->GetInt()), &version, &attach);
AddLine(wxString::Format("PlayResY: %" PRId64, OPT_GET("Subtitle/Default Resolution/Height")->GetInt()), &version, &attach); AddLine(wxString::Format("PlayResY: %" PRId64, OPT_GET("Subtitle/Default Resolution/Height")->GetInt()), &version, &attach);
} }
AddLine("YCbCr Matrix: None", &version, &attach);
InsertStyle(new AssStyle); InsertStyle(new AssStyle);

View file

@ -65,9 +65,9 @@ public:
virtual std::vector<int> GetKeyFrames() const=0;///< Returns list of keyframes virtual std::vector<int> GetKeyFrames() const=0;///< Returns list of keyframes
/// Get the source colorspace of the video before it was converted to RGB /// Get the source colorspace of the video before it was converted to RGB
/// @return A string describing the source colorspace or empty if it is /// @return A string describing the source colorspace or "None" if it is
/// unknown or meaningless /// unknown or meaningless
virtual wxString GetColorSpace() const { return ""; } virtual wxString GetColorSpace() const = 0;
/// @brief Use this to set any post-loading warnings, such as "being loaded with unreliable seeking" /// @brief Use this to set any post-loading warnings, such as "being loaded with unreliable seeking"
virtual wxString GetWarning() const { return ""; } virtual wxString GetWarning() const { return ""; }

View file

@ -250,7 +250,6 @@ void VideoContext::OnSubtitlesCommit() {
void VideoContext::OnSubtitlesSave() { void VideoContext::OnSubtitlesSave() {
if (!IsLoaded()) { if (!IsLoaded()) {
context->ass->SetScriptInfo("Video File", ""); context->ass->SetScriptInfo("Video File", "");
context->ass->SetScriptInfo("Video Colorspace", "");
context->ass->SetScriptInfo("Video Aspect Ratio", ""); context->ass->SetScriptInfo("Video Aspect Ratio", "");
context->ass->SetScriptInfo("Video Position", ""); context->ass->SetScriptInfo("Video Position", "");
context->ass->SetScriptInfo("VFR File", ""); context->ass->SetScriptInfo("VFR File", "");
@ -265,7 +264,7 @@ void VideoContext::OnSubtitlesSave() {
ar = wxString::Format("%d", arType); ar = wxString::Format("%d", arType);
context->ass->SetScriptInfo("Video File", MakeRelativePath(videoFile, context->ass->filename)); context->ass->SetScriptInfo("Video File", MakeRelativePath(videoFile, context->ass->filename));
context->ass->SetScriptInfo("Video Colorspace", videoProvider->GetColorSpace()); context->ass->SetScriptInfo("YCbCr Matrix", videoProvider->GetColorSpace());
context->ass->SetScriptInfo("Video Aspect Ratio", ar); context->ass->SetScriptInfo("Video Aspect Ratio", ar);
context->ass->SetScriptInfo("Video Position", wxString::Format("%d", frame_n)); context->ass->SetScriptInfo("Video Position", wxString::Format("%d", frame_n));
context->ass->SetScriptInfo("VFR File", MakeRelativePath(GetTimecodesName(), context->ass->filename)); context->ass->SetScriptInfo("VFR File", MakeRelativePath(GetTimecodesName(), context->ass->filename));

View file

@ -162,18 +162,15 @@ file_exit:
AVSValue args[2] = { script, "Rec601" }; AVSValue args[2] = { script, "Rec601" };
if (!OPT_GET("Video/Force BT.601")->GetBool() && (vi.width > 1024 || vi.height >= 600)) { if (!OPT_GET("Video/Force BT.601")->GetBool() && (vi.width > 1024 || vi.height >= 600)) {
args[1] = "Rec709"; args[1] = "Rec709";
colorspace = "BT.709"; colorspace = "TV.709";
} }
else else
colorspace = "BT.601"; colorspace = "TV.601";
const char *argnames[2] = { 0, "matrix" }; const char *argnames[2] = { 0, "matrix" };
script = avs.GetEnv()->Invoke("ConvertToRGB32", AVSValue(args, 2), argnames); script = avs.GetEnv()->Invoke("ConvertToRGB32", AVSValue(args, 2), argnames);
} }
else if (extension != ".avs") else
colorspace = "RGB"; colorspace = "None";
// Don't set the colorspace to RGB if we're opening an Avisynth script
// as we can't tell RGB source video apart from a script that happens
// to convert to rgb
RGB32Video = avs.GetEnv()->Invoke("Cache", script).AsClip(); RGB32Video = avs.GetEnv()->Invoke("Cache", script).AsClip();
vi = RGB32Video->GetVideoInfo(); vi = RGB32Video->GetVideoInfo();

View file

@ -90,6 +90,7 @@ public:
int GetHeight() const { return height; } int GetHeight() const { return height; }
double GetDAR() const { return 0; } double GetDAR() const { return 0; }
agi::vfr::Framerate GetFPS() const { return fps; } agi::vfr::Framerate GetFPS() const { return fps; }
std::vector<int> GetKeyFrames() const { return std::vector<int>(); }; std::vector<int> GetKeyFrames() const { return std::vector<int>(); }
wxString GetColorSpace() const { return "None"; }
wxString GetDecoderName() const { return "Dummy Video Provider"; } wxString GetDecoderName() const { return "Dummy Video Provider"; }
}; };

View file

@ -179,6 +179,9 @@ void FFmpegSourceVideoProvider::LoadVideo(wxString filename) {
else else
DAR = double(Width) / Height; DAR = double(Width) / Height;
// Assuming TV for unspecified
wxString ColorRange = TempFrame->ColorRange == FFMS_CR_JPEG ? "PC" : "TV";
int CS = TempFrame->ColorSpace; int CS = TempFrame->ColorSpace;
#if FFMS_VERSION >= ((2 << 24) | (17 << 16) | (1 << 8) | 0) #if FFMS_VERSION >= ((2 << 24) | (17 << 16) | (1 << 8) | 0)
if (CS != FFMS_CS_RGB && CS != FFMS_CS_BT470BG && OPT_GET("Video/Force BT.601")->GetBool()) { if (CS != FFMS_CS_RGB && CS != FFMS_CS_BT470BG && OPT_GET("Video/Force BT.601")->GetBool()) {
@ -186,15 +189,33 @@ void FFmpegSourceVideoProvider::LoadVideo(wxString filename) {
throw VideoOpenError(std::string("Failed to set input format: ") + ErrInfo.Buffer); throw VideoOpenError(std::string("Failed to set input format: ") + ErrInfo.Buffer);
CS = FFMS_CS_BT470BG; CS = FFMS_CS_BT470BG;
ColorRange = "TV";
} }
#endif #endif
switch (CS) { switch (CS) {
case FFMS_CS_RGB: ColorSpace = "RGB"; break; case FFMS_CS_RGB:
case FFMS_CS_BT709: ColorSpace = "BT.709"; break; ColorSpace = "None";
case FFMS_CS_BT470BG: ColorSpace = "BT.601"; break; break;
case FFMS_CS_UNSPECIFIED: ColorSpace = Width > 1024 || Height >= 600 ? "BT.709" : "BT.601"; break; case FFMS_CS_BT709:
default: ColorSpace = ""; break; ColorSpace = wxString::Format("%s.709", ColorRange);
break;
case FFMS_CS_UNSPECIFIED:
ColorSpace = wxString::Format("%s.%s", ColorRange, Width > 1024 || Height >= 600 ? "709" : "601");
break;
case FFMS_CS_FCC:
ColorSpace = wxString::Format("%s.FCC", ColorRange);
break;
case FFMS_CS_BT470BG:
case FFMS_CS_SMPTE170M:
ColorSpace = wxString::Format("%s.601", ColorRange);
break;
case FFMS_CS_SMPTE240M:
ColorSpace = wxString::Format("%s.204M", ColorRange);
break;
default:
throw VideoOpenError("Unknown video color space");
break;
} }
const int TargetFormat[] = { FFMS_GetPixFmt("bgra"), -1 }; const int TargetFormat[] = { FFMS_GetPixFmt("bgra"), -1 };

View file

@ -148,7 +148,8 @@ public:
int GetHeight() const { return h; } int GetHeight() const { return h; }
double GetDAR() const { return 0; } double GetDAR() const { return 0; }
agi::vfr::Framerate GetFPS() const { return fps; } agi::vfr::Framerate GetFPS() const { return fps; }
std::vector<int> GetKeyFrames() const { return std::vector<int>(); }; std::vector<int> GetKeyFrames() const { return std::vector<int>(); }
wxString GetDecoderName() const { return "YU4MPEG"; }; wxString GetColorSpace() const { return "TV.601"; }
bool WantsCaching() const { return true; }; wxString GetDecoderName() const { return "YU4MPEG"; }
bool WantsCaching() const { return true; }
}; };