Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
V
vlc-2-2
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Redmine
Redmine
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Operations
Operations
Metrics
Environments
Analytics
Analytics
CI / CD
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
videolan
vlc-2-2
Commits
99bc1b47
Commit
99bc1b47
authored
Apr 19, 2011
by
Juha Jeronen
Committed by
Jean-Baptiste Kempf
Apr 19, 2011
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
IVTC trivial fixes 1
Signed-off-by:
Jean-Baptiste Kempf
<
jb@videolan.org
>
parent
960d8fcd
Changes
1
Hide whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
160 additions
and
145 deletions
+160
-145
modules/video_filter/deinterlace.c
modules/video_filter/deinterlace.c
+160
-145
No files found.
modules/video_filter/deinterlace.c
View file @
99bc1b47
...
@@ -3032,13 +3032,13 @@ static int RenderPhosphor( filter_t *p_filter,
...
@@ -3032,13 +3032,13 @@ static int RenderPhosphor( filter_t *p_filter,
* every chroma line, even if the chroma format is 4:2:0!
* every chroma line, even if the chroma format is 4:2:0!
*
*
* This is just a raw detector that produces a score. The overall score
* This is just a raw detector that produces a score. The overall score
* indicating a progressive or in
dicat
ed frame may vary wildly, depending on
* indicating a progressive or in
terlac
ed frame may vary wildly, depending on
* the material, especially in anime. The scores should be compared to
* the material, especially in anime. The scores should be compared to
* each other locally (in the temporal sense) to make meaningful decisions
* each other locally (in the temporal sense) to make meaningful decisions
* about progressive or interlaced frames.
* about progressive or interlaced frames.
*
*
* @param p_pic_top Picture to take the top field from.
* @param p_pic_top Picture to take the top field from.
* @param p_pic_bot Picture to take the bottom field from.
* @param p_pic_bot Picture to take the bottom field from
(same or different)
.
* @return Interlace score, >= 0. Higher values mean more interlaced.
* @return Interlace score, >= 0. Higher values mean more interlaced.
* @retval -1 Error: incompatible input pictures.
* @retval -1 Error: incompatible input pictures.
* @see RenderIVTC()
* @see RenderIVTC()
...
@@ -3363,13 +3363,17 @@ static inline int TestForMotionInBlock( uint8_t *p_pix_p, uint8_t *p_pix_c,
...
@@ -3363,13 +3363,17 @@ static inline int TestForMotionInBlock( uint8_t *p_pix_p, uint8_t *p_pix_c,
* Note that the return value is NOT simply *pi_top + *pi_bot, because
* Note that the return value is NOT simply *pi_top + *pi_bot, because
* the fields and the full block use different motion thresholds.
* the fields and the full block use different motion thresholds.
*
*
* I
t is allowed to set pi_top and pi_bot to NULL, if the caller does not want
* I
f you do not want the separate field scores, pass NULL for pi_top and
*
the separate field scores. This does not affect computation speed, and i
s
*
pi_bot. This does not affect computation speed, and is only provided a
s
*
only provided as
a syntactic convenience.
* a syntactic convenience.
*
*
* Motion in each picture plane (Y, U, V) counts separately.
* Motion in each picture plane (Y, U, V) counts separately.
* The sum of number of blocks with motion across all planes is returned.
* The sum of number of blocks with motion across all planes is returned.
*
*
* For 4:2:0 chroma, even-numbered chroma lines make up the "top field" for
* chroma, and odd-numbered chroma lines the "bottom field" for chroma.
* This is correct for IVTC purposes.
*
* @param[in] p_prev Previous picture
* @param[in] p_prev Previous picture
* @param[in] p_curr Current picture
* @param[in] p_curr Current picture
* @param[out] pi_top Number of 8x8 blocks where top field has motion.
* @param[out] pi_top Number of 8x8 blocks where top field has motion.
...
@@ -3504,7 +3508,10 @@ typedef enum { CADENCE_POS_INVALID = -1,
...
@@ -3504,7 +3508,10 @@ typedef enum { CADENCE_POS_INVALID = -1,
#define CADENCE_POS_BFF_END 9
#define CADENCE_POS_BFF_END 9
/**
/**
* For Vektor-like cadence detector algorithm.
* For the "vektor" cadence detector algorithm.
*
* The algorithm produces a set of possible positions instead of a unique
* position, until it locks on. The set is represented as a bitmask.
*
*
* The bitmask is stored in a word, and its layout is:
* The bitmask is stored in a word, and its layout is:
* blank blank BFF_CARRY BFF4 BFF3 BFF2 BFF1 BFF0 (high byte)
* blank blank BFF_CARRY BFF4 BFF3 BFF2 BFF1 BFF0 (high byte)
...
@@ -3514,7 +3521,8 @@ typedef enum { CADENCE_POS_INVALID = -1,
...
@@ -3514,7 +3521,8 @@ typedef enum { CADENCE_POS_INVALID = -1,
* result by one bit, copying the CARRY bits to the respective zeroth position,
* result by one bit, copying the CARRY bits to the respective zeroth position,
* and ANDing with 0x1F1F.
* and ANDing with 0x1F1F.
*
*
* The table is indexed with a valid ivtc_cadence_pos.
* This table is indexed with a valid ivtc_cadence_pos.
* @see ivtc_cadence_pos
*/
*/
const
int
pi_detected_pos_to_bitmask
[
NUM_CADENCE_POS
]
=
{
0x0808
,
/* prog. */
const
int
pi_detected_pos_to_bitmask
[
NUM_CADENCE_POS
]
=
{
0x0808
,
/* prog. */
0x0001
,
/* TFF ABC */
0x0001
,
/* TFF ABC */
...
@@ -3538,12 +3546,12 @@ const int pi_detected_pos_to_bitmask[NUM_CADENCE_POS] = { 0x0808, /* prog. */
...
@@ -3538,12 +3546,12 @@ const int pi_detected_pos_to_bitmask[NUM_CADENCE_POS] = { 0x0808, /* prog. */
typedef
enum
{
TFD_INVALID
=
-
1
,
TFD_TFF
=
0
,
TFD_BFF
=
1
}
ivtc_tfd
;
typedef
enum
{
TFD_INVALID
=
-
1
,
TFD_TFF
=
0
,
TFD_BFF
=
1
}
ivtc_tfd
;
/**
/**
* Position detection table.
* Position detection table
for the "scores" cadence detector algorithm
.
*
*
* These are the (only) field pair combinations that should give progressive
* These are the (only) field pair combinations that should give progressive
* frames.
* frames.
There are three for each position.
*
*
* First index:
detected
pos
* First index:
ivtc_cadence_
pos
*/
*/
static
const
ivtc_field_pair
pi_best_field_pairs
[
NUM_CADENCE_POS
][
3
]
=
{
static
const
ivtc_field_pair
pi_best_field_pairs
[
NUM_CADENCE_POS
][
3
]
=
{
{
FIELD_PAIR_TPBP
,
FIELD_PAIR_TCBC
,
FIELD_PAIR_TNBN
},
/* prog. */
{
FIELD_PAIR_TPBP
,
FIELD_PAIR_TCBC
,
FIELD_PAIR_TNBN
},
/* prog. */
...
@@ -3560,9 +3568,13 @@ static const ivtc_field_pair pi_best_field_pairs[NUM_CADENCE_POS][3] = {
...
@@ -3560,9 +3568,13 @@ static const ivtc_field_pair pi_best_field_pairs[NUM_CADENCE_POS][3] = {
};
};
/**
/**
* Alternative position detection table.
* Alternative position detection table for the "scores" cadence detector
* algorithm.
*
*
* These field pair combinations should give only interlaced frames.
* These field pair combinations should give only interlaced frames.
* There are four for each position.
*
* First index: ivtc_cadence_pos
*
*
* Currently unused. During development it was tested that whether we detect
* Currently unused. During development it was tested that whether we detect
* best or worst, the resulting detected cadence positions are identical
* best or worst, the resulting detected cadence positions are identical
...
@@ -3635,7 +3647,8 @@ static const int pi_detected_pos_to_tfd[NUM_CADENCE_POS] = {
...
@@ -3635,7 +3647,8 @@ static const int pi_detected_pos_to_tfd[NUM_CADENCE_POS] = {
/* Valid telecine sequences (TFF and BFF). Indices: [TFD][i_cadence_pos] */
/* Valid telecine sequences (TFF and BFF). Indices: [TFD][i_cadence_pos] */
/* Currently unused and left here for documentation only.
/* Currently unused and left here for documentation only.
There is an easier way - just decode the i_cadence_pos part of the
There is an easier way - just decode the i_cadence_pos part of the
detected position using the pi_detected_pos_to_cadence_pos table. */
detected position using the pi_detected_pos_to_cadence_pos table,
and check that it is successive mod 5. See IVTCCadenceAnalyze(). */
/*static const int pi_valid_cadences[2][5] = { {CADENCE_POS_TFF_ABC,
/*static const int pi_valid_cadences[2][5] = { {CADENCE_POS_TFF_ABC,
CADENCE_POS_TFF_BCD,
CADENCE_POS_TFF_BCD,
CADENCE_POS_TFF_CDE,
CADENCE_POS_TFF_CDE,
...
@@ -3694,7 +3707,7 @@ static const ivtc_op pi_reconstruction_ops[2][5] = { /* TFF */
...
@@ -3694,7 +3707,7 @@ static const ivtc_op pi_reconstruction_ops[2][5] = { /* TFF */
*
*
* This is used in the 29.97 -> 23.976 fps conversion.
* This is used in the 29.97 -> 23.976 fps conversion.
*
*
* Index: i_cadence_pos.
* Index: i_cadence_pos
, 0..4
.
*
*
* Valid values are nonnegative. The -1 corresponds to the dropped frame
* Valid values are nonnegative. The -1 corresponds to the dropped frame
* and is never used, except for a debug assert.
* and is never used, except for a debug assert.
...
@@ -3715,7 +3728,8 @@ static const int pi_timestamp_deltas[5] = { 1, 2, 3, -1, 0 };
...
@@ -3715,7 +3728,8 @@ static const int pi_timestamp_deltas[5] = { 1, 2, 3, -1, 0 };
* In practice, this slides detector histories.
* In practice, this slides detector histories.
*
*
* This function should only perform initialization that does NOT require
* This function should only perform initialization that does NOT require
* the input frame history buffer.
* the input frame history buffer. This runs at every frame, including
* the first two.
*
*
* This is an internal function only used by RenderIVTC().
* This is an internal function only used by RenderIVTC().
* There is no need to call this function manually.
* There is no need to call this function manually.
...
@@ -3765,6 +3779,10 @@ static inline void IVTCFrameInit( filter_t *p_filter )
...
@@ -3765,6 +3779,10 @@ static inline void IVTCFrameInit( filter_t *p_filter )
p_ivtc
->
pi_scores
[
FIELD_PAIR_TPBC
]
=
p_ivtc
->
pi_scores
[
FIELD_PAIR_TCBN
];
p_ivtc
->
pi_scores
[
FIELD_PAIR_TPBC
]
=
p_ivtc
->
pi_scores
[
FIELD_PAIR_TCBN
];
p_ivtc
->
pi_scores
[
FIELD_PAIR_TCBP
]
=
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBC
];
p_ivtc
->
pi_scores
[
FIELD_PAIR_TCBP
]
=
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBC
];
p_ivtc
->
pi_scores
[
FIELD_PAIR_TCBC
]
=
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBN
];
p_ivtc
->
pi_scores
[
FIELD_PAIR_TCBC
]
=
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBN
];
/* These have not been detected yet */
p_ivtc
->
pi_scores
[
FIELD_PAIR_TCBN
]
=
0
;
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBC
]
=
0
;
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBN
]
=
0
;
}
}
/**
/**
...
@@ -3807,9 +3825,8 @@ static inline void IVTCLowLevelDetect( filter_t *p_filter )
...
@@ -3807,9 +3825,8 @@ static inline void IVTCLowLevelDetect( filter_t *p_filter )
int
i_motion
=
EstimateNumBlocksWithMotion
(
p_curr
,
p_next
,
&
i_top
,
&
i_bot
);
int
i_motion
=
EstimateNumBlocksWithMotion
(
p_curr
,
p_next
,
&
i_top
,
&
i_bot
);
p_ivtc
->
pi_motion
[
IVTC_LATEST
]
=
i_motion
;
p_ivtc
->
pi_motion
[
IVTC_LATEST
]
=
i_motion
;
/* It's very rare if nonexistent that only one field changes between
/* If one field changes "clearly more" than the other, we know the
frames. Thus, if one field changes "clearly more" than the other,
less changed one is a likely duplicate.
we know the less changed one is a likely duplicate.
Threshold 1/2 is too low for some scenes (e.g. pan of the space junk
Threshold 1/2 is too low for some scenes (e.g. pan of the space junk
at beginning of The Third ep. 1, right after the OP). Thus, we use 2/3,
at beginning of The Third ep. 1, right after the OP). Thus, we use 2/3,
...
@@ -3821,7 +3838,7 @@ static inline void IVTCLowLevelDetect( filter_t *p_filter )
...
@@ -3821,7 +3838,7 @@ static inline void IVTCLowLevelDetect( filter_t *p_filter )
/**
/**
* Internal helper function for RenderIVTC(): using raw detector data,
* Internal helper function for RenderIVTC(): using raw detector data,
* detect cadence position by an interlace scores based algorithm.
* detect cadence position by an interlace scores based algorithm
("scores")
.
*
*
* IVTCFrameInit() and IVTCLowLevelDetect() must have been called first.
* IVTCFrameInit() and IVTCLowLevelDetect() must have been called first.
* Last frame must be available in the history buffer.
* Last frame must be available in the history buffer.
...
@@ -3856,9 +3873,8 @@ static inline void IVTCCadenceDetectAlgoScores( filter_t *p_filter )
...
@@ -3856,9 +3873,8 @@ static inline void IVTCCadenceDetectAlgoScores( filter_t *p_filter )
/* Find minimum */
/* Find minimum */
int
j
=
CADENCE_POS_PROGRESSIVE
;
/* valid regardless of TFD */
int
j
=
CADENCE_POS_PROGRESSIVE
;
/* valid regardless of TFD */
int
minscore
=
pi_ivtc_scores
[
j
];
int
minscore
=
pi_ivtc_scores
[
j
];
/* Note that a TFF (respectively BFF) stream may only have TFF
/* A TFF (respectively BFF) stream may only have TFF (respectively BFF)
(respectively BFF) telecine. Don't bother looking at solutions
telecine. Don't bother looking at the wrong table. */
we already know to be wrong. */
int
imin
=
CADENCE_POS_TFF_FIRST
;
/* first TFF-only entry */
int
imin
=
CADENCE_POS_TFF_FIRST
;
/* first TFF-only entry */
int
iend
=
CADENCE_POS_TFF_END
;
/* one past last TFF-only entry */
int
iend
=
CADENCE_POS_TFF_END
;
/* one past last TFF-only entry */
if
(
!
p_next
->
b_top_field_first
)
if
(
!
p_next
->
b_top_field_first
)
...
@@ -3909,13 +3925,13 @@ static inline void IVTCCadenceDetectAlgoScores( filter_t *p_filter )
...
@@ -3909,13 +3925,13 @@ static inline void IVTCCadenceDetectAlgoScores( filter_t *p_filter )
double
mean_ratio
=
(
double
)
mean_except_min
/
(
double
)
mean
;
double
mean_ratio
=
(
double
)
mean_except_min
/
(
double
)
mean
;
/* Let's pretend that the detected position is a stochastic variable.
/* Let's pretend that the detected position is a stochastic variable.
Compute sample variance with the winner included and without.
Compute sample variance with the winner included and without.
var = sum( (x_i - mu)^2, i ) / N ,
var = sum( (x_i - mu)^2, i ) / N ,
where mu is the sample mean.
where mu is the sample mean.
Note that we really need int64_t; the numbers are pretty large.
Note that we really need int64_t; the numbers are pretty large.
*/
*/
int64_t
diff
=
(
int64_t
)(
pi_ivtc_scores
[
CADENCE_POS_PROGRESSIVE
]
-
mean
);
int64_t
diff
=
(
int64_t
)(
pi_ivtc_scores
[
CADENCE_POS_PROGRESSIVE
]
-
mean
);
int64_t
var
=
diff
*
diff
;
int64_t
var
=
diff
*
diff
;
...
@@ -3991,7 +4007,7 @@ static inline void IVTCCadenceDetectAlgoScores( filter_t *p_filter )
...
@@ -3991,7 +4007,7 @@ static inline void IVTCCadenceDetectAlgoScores( filter_t *p_filter )
/**
/**
* Internal helper function for RenderIVTC(): using raw detector data,
* Internal helper function for RenderIVTC(): using raw detector data,
* detect cadence position by a hard field repeat based algorithm.
* detect cadence position by a hard field repeat based algorithm
("vektor")
.
*
*
* This algorithm is inspired by the classic TVTime/Xine IVTC filter
* This algorithm is inspired by the classic TVTime/Xine IVTC filter
* by Billy Biggs (Vektor); hence the name. There are however some
* by Billy Biggs (Vektor); hence the name. There are however some
...
@@ -4020,46 +4036,55 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
...
@@ -4020,46 +4036,55 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
assert
(
p_next
!=
NULL
);
assert
(
p_next
!=
NULL
);
/* Vektor-like cadence detection algorithm.
/* This algorithm is based on detecting hard-repeated fields (by motion
detection), and conservatively estimating what the seen repeats could
mean for the cadence position.
This is based on detecting repeated fields (by motion detection),
"Conservative" means that we do not rule out possibilities if repeats
and conservatively estimating what the seen repeats could mean
are *not* seen, but only *add* possibilities based on what repeats
for the cadence position.
*are* seen. This is important. Otherwise full-frame repeats in the
original film (8fps or 12fps animation is very common in anime),
causing spurious field repeats, would mess up the detection.
With this strategy, spurious repeats will only slow down the lock-on,
and will not break an existing lock-on once acquired.
Several possibilities are kept open until the sequence gives enough
Several possibilities are kept open until the sequence gives enough
information to make a unique detection. When the sequence becomes
information to make a unique detection. When the sequence becomes
inconsistent (e.g. bad cut), the detector resets itself.
inconsistent (e.g. bad cut), the detector resets itself.
The main ideas taken from
Vektor's
algorithm are:
The main ideas taken from
the TVTime/Xine
algorithm are:
1)
c
onservatively using information from detected field repeats,
1)
C
onservatively using information from detected field repeats,
2)
c
adence counting the earlier detection results and combining with
2)
C
adence counting the earlier detection results and combining with
the new detection result, and
the new detection result, and
3)
t
he observation that video TFF/BFF uniquely determines TFD.
3)
T
he observation that video TFF/BFF uniquely determines TFD.
The main differences are
The main differences are
1)
d
ifferent motion detection (see EstimateNumBlocksWithMotion()).
1)
D
ifferent motion detection (see EstimateNumBlocksWithMotion()).
Vektor's original estimates the average top/bottom field diff
Vektor's original estimates the average top/bottom field diff
over the last 3 frames, while ours uses a block-based approach
over the last 3 frames, while ours uses a block-based approach
for diffing and just compares the field diffs
of the "next" frame
for diffing and just compares the field diffs
between "curr" and
against each other. Both approaches are adaptive, but in a
"next" against each other (see IVTCLowLevelDetect()).
different way.
Both approaches are adaptive, but in a
different way.
2)
the specific detection logic used is a bit different (see both codes
2)
The specific detection logic used is a bit different (see both
for details; the original is in xine-lib, function
codes
for details; the original is in xine-lib, function
determine_pulldown_offset_short_history_new() in pulldown.c;
determine_pulldown_offset_short_history_new() in pulldown.c;
ours is obviously given below). I think th
is one is a bit simpler.
ours is obviously given below). I think th
e one given here
*/
is a bit simpler.
bool
b_top_rep
=
p_ivtc
->
pi_top_rep
[
IVTC_LATEST
];
Note that we don't have to worry about getting a detection in all cases.
bool
b_bot_rep
=
p_ivtc
->
pi_bot_rep
[
IVTC_LATEST
];
It's enough if we work reliably, say, 99% of the time, and the other 1%
bool
b_old_top_rep
=
p_ivtc
->
pi_top_rep
[
IVTC_LATEST
-
1
];
of the time just admit that we don't know the cadence position.
bool
b_old_bot_rep
=
p_ivtc
->
pi_bot_rep
[
IVTC_LATEST
-
1
];
(This mostly happens after a bad cut, when the new scene has
"difficult" motion characteristics, such as repeated film frames.)
/* This is a conservative algorithm: we do not rule out possibilities
Our frame composer is built to handle also cases where we have no
if repeats are *not* seen, but only *add* possibilities based on what
reliable detection of the cadence position; see IVTCOutputOrDropFrame().
repeats *are* seen. We will do a raw detection, whose result is then
More important is to never lock on incorrectly, as this would both
filtered against what we already know.
generate interlacing artifacts where none existed, and cause motion
to stutter (because duplicate frames would be shown and unique ones
dropped).
*/
Progressive requires no repeats, so it is always a possibility.
/*
Progressive requires no repeats, so it is always a possibility.
Filtering will drop it out if we know that the current position
Filtering will drop it out if we know that the current position
cannot be "dea".
cannot be "dea".
*/
*/
...
@@ -4069,34 +4094,11 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
...
@@ -4069,34 +4094,11 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
/* Add in other possibilities depending on field repeats seen during the
/* Add in other possibilities depending on field repeats seen during the
last three input frames (i.e. two transitions between input frames).
last three input frames (i.e. two transitions between input frames).
See the "Dups." column in the cadence tables.
See the "Dups." column in the cadence tables.
Note that we always add and never explicitly rule anything out.
This is important. Otherwise full-frame repeats in the original film
(8fps or 12fps animation is common in anime) - causing spurious
field repeats - would mess up the detection. Handling that in a more
sophisticated way would be a nightmare - one would have to keep track
of full-frame repeats in the *outgoing* frames, too, and take into
account what would happen in the output if a particular cadence position
was chosen. Accounting for repeats in input frames only (i.e. limiting
the detection to the progressive parts of the cadence), this has been
tried, and found less reliable than the current, simpler strategy
that just ignores full-frame repeats.
Note also that we don't have to worry about getting the detection right
in *all* cases. It's enough if we work reliably, say, 99% of the time,
and the other 1% of the time just admit that we don't know the cadence
position. (This mostly happens after a bad cut, when the new scene has
"difficult" motion characteristics, such as repeated film frames.)
The alternative, "Transcode" strategy in the frame composer will catch
any telecined frames that slip through. Although in that case there will
be duplicates and the output PTSs will be wrong, this is less noticeable
than getting PTS jumps from an incorrectly locked-on cadence. Note that
it is mostly anime, and even there mostly low-motion scenes with
duplicate film frames that trigger the misbehavior - and in such cases
any slight irregularity in the output timings will go unnoticed,
as long as we get rid of interlacing artifacts.
*/
*/
bool
b_top_rep
=
p_ivtc
->
pi_top_rep
[
IVTC_LATEST
];
bool
b_bot_rep
=
p_ivtc
->
pi_bot_rep
[
IVTC_LATEST
];
bool
b_old_top_rep
=
p_ivtc
->
pi_top_rep
[
IVTC_LATEST
-
1
];
bool
b_old_bot_rep
=
p_ivtc
->
pi_bot_rep
[
IVTC_LATEST
-
1
];
if
(
b_top_rep
)
if
(
b_top_rep
)
{
{
detected
|=
pi_detected_pos_to_bitmask
[
CADENCE_POS_TFF_EAB
];
detected
|=
pi_detected_pos_to_bitmask
[
CADENCE_POS_TFF_EAB
];
...
@@ -4119,10 +4121,10 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
...
@@ -4119,10 +4121,10 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
}
}
/* A TFF stream may only have TFF telecine, and similarly for BFF.
/* A TFF stream may only have TFF telecine, and similarly for BFF.
Discard the possibility we know to be incorrect for this stream.
Discard the possibility we know to be incorrect for this stream.
(Note that the stream may flipflop between the possibilitie
s
(The stream may flipflop between the possibilities if it contain
s
if it contains soft-telecined sequences or lone field repeats,
soft-telecined sequences or lone field repeats, so we must keep
so we must keep
detecting this for each incoming frame.)
detecting this for each incoming frame.)
*/
*/
bool
b_tff
=
p_next
->
b_top_field_first
;
bool
b_tff
=
p_next
->
b_top_field_first
;
if
(
b_tff
)
if
(
b_tff
)
...
@@ -4149,7 +4151,8 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
...
@@ -4149,7 +4151,8 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
/* Filter: narrow down possibilities based on previous detection,
/* Filter: narrow down possibilities based on previous detection,
if consistent. If not consistent, reset the detector.
if consistent. If not consistent, reset the detector.
This works better than just using the latest raw detection. */
Reset works better than just using the latest raw detection.
*/
if
(
(
detected
&
predicted
)
!=
0
)
if
(
(
detected
&
predicted
)
!=
0
)
detected
=
detected
&
predicted
;
detected
=
detected
&
predicted
;
else
else
...
@@ -4158,9 +4161,9 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
...
@@ -4158,9 +4161,9 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
/* We're done. Save result to our internal storage so we can use it
/* We're done. Save result to our internal storage so we can use it
for prediction at the next frame.
for prediction at the next frame.
Note that the outgoing frame check in IVTC
Reconstruct
Frame()
Note that the outgoing frame check in IVTC
OutputOrDrop
Frame()
has a veto right, resetting
us if it determines that the cadence
has a veto right, resetting
our state if it determines that
has become broken.
the cadence
has become broken.
*/
*/
p_ivtc
->
pi_v_raw
[
IVTC_LATEST
]
=
detected
;
p_ivtc
->
pi_v_raw
[
IVTC_LATEST
]
=
detected
;
...
@@ -4194,7 +4197,7 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
...
@@ -4194,7 +4197,7 @@ static inline void IVTCCadenceDetectAlgoVektor( filter_t *p_filter )
/**
/**
* Internal helper function for RenderIVTC(): decide the final detected
* Internal helper function for RenderIVTC(): decide the final detected
* cadence position for the current position of the stencil,
* cadence position for the current position of the
PCN
stencil,
* using the results of the different cadence detection algorithms.
* using the results of the different cadence detection algorithms.
*
*
* Must be called after all IVTCCadenceDetectAlgo*() functions.
* Must be called after all IVTCCadenceDetectAlgo*() functions.
...
@@ -4218,7 +4221,8 @@ static inline void IVTCCadenceDetectFinalize( filter_t *p_filter )
...
@@ -4218,7 +4221,8 @@ static inline void IVTCCadenceDetectFinalize( filter_t *p_filter )
take longer to lock on. Thus, we prefer "vektor" if its reliable bit
take longer to lock on. Thus, we prefer "vektor" if its reliable bit
is set, then "scores", and finally just give up.
is set, then "scores", and finally just give up.
For progressive sequences, "vektor" outputs "3, -, 3, -, ...".
For progressive sequences, "vektor" outputs "3, -, 3, -, ...",
because the repeated progressive position is an inconsistent prediction.
In this case, "scores" fills in the blanks. (This particular task
In this case, "scores" fills in the blanks. (This particular task
could also be done without another cadence detector, by just
could also be done without another cadence detector, by just
detecting the alternating pattern of "3" and no result.)
detecting the alternating pattern of "3" and no result.)
...
@@ -4268,16 +4272,16 @@ static inline void IVTCSoftTelecineDetect( filter_t *p_filter )
...
@@ -4268,16 +4272,16 @@ static inline void IVTCSoftTelecineDetect( filter_t *p_filter )
nb_fields = 3,2,3,2,... and *video* TFF = true, false, false, true
nb_fields = 3,2,3,2,... and *video* TFF = true, false, false, true
(TFF telecine) or false, true, true, false (BFF telecine).
(TFF telecine) or false, true, true, false (BFF telecine).
We don't particularly care which field goes first, because
we're
We don't particularly care which field goes first, because
in soft TC
constructing progressive frames, and the video FDs of successive frame
s
we're working with progressive frames. And in any case, the video FD
s
must in any case match any field repeats in order for field renderers
of successive frames must match any field repeats in order for field
(such as traditional DVD player + CRT TV) to work correctly. Thus the
renderers (such as traditional DVD player + CRT TV) to work correctly.
video TFF/BFF flag provides no additional useful information for us
Thus the video TFF/BFF flag provides no additional useful information
on top of checking nb_fields.
for us
on top of checking nb_fields.
Note that the only thing to *do* to soft telecine in an IVTC filter
The only thing to *do* to soft telecine in an IVTC filter is to even
is to even
out the outgoing PTS diffs to 2.5 fields each, so that we get
out the outgoing PTS diffs to 2.5 fields each, so that we get
a steady 24fps output. Thus, we can do this
processing even if it turns
a steady 24fps output. Thus, we can do this processing even if it turns
out that we saw a lone field repeat (which are also sometimes used,
out that we saw a lone field repeat (which are also sometimes used,
such as in the Silent Mobius OP and in Sol Bianca). We can be aggressive
such as in the Silent Mobius OP and in Sol Bianca). We can be aggressive
and don't need to care about false positives - as long as we are equally
and don't need to care about false positives - as long as we are equally
...
@@ -4286,8 +4290,8 @@ static inline void IVTCSoftTelecineDetect( filter_t *p_filter )
...
@@ -4286,8 +4290,8 @@ static inline void IVTCSoftTelecineDetect( filter_t *p_filter )
Finally, we conclude that the one-frame future buffer is enough for us
Finally, we conclude that the one-frame future buffer is enough for us
to make soft TC decisions just in time for rendering the frame in the
to make soft TC decisions just in time for rendering the frame in the
"current" position
(the flag patterns below constitute proof of this
"current" position
. The flag patterns given below constitute proof
property)
.
of this property
.
Soft telecine is relatively rare at least in anime, but it exists;
Soft telecine is relatively rare at least in anime, but it exists;
e.g. Angel Links OP, Silent Mobius, and Stellvia of the Universe have
e.g. Angel Links OP, Silent Mobius, and Stellvia of the Universe have
...
@@ -4339,15 +4343,16 @@ static inline void IVTCSoftTelecineDetect( filter_t *p_filter )
...
@@ -4339,15 +4343,16 @@ static inline void IVTCSoftTelecineDetect( filter_t *p_filter )
/* No longer soft telecined, return filter to the mode it had earlier.
/* No longer soft telecined, return filter to the mode it had earlier.
This is needed to fix cases where we came in from hard telecine, and
This is needed to fix cases where we came in from hard telecine, and
should go back, but can't catch a cadence in time before telecined
should go back, but can't catch a cadence in time before telecined
frames slip through. Kickstarting back to hard IVTC fixes the
frames slip through. Kickstarting back to hard IVTC, using the
problem. This happens a lot in Stellvia.
emergency frame composer until the cadence locks on again,
fixes the problem. This happens a lot in Stellvia.
*/
*/
p_ivtc
->
i_mode
=
p_ivtc
->
i_old_mode
;
p_ivtc
->
i_mode
=
p_ivtc
->
i_old_mode
;
p_ivtc
->
i_cadence_pos
=
0
;
/* Wild guess. The film frame reconstruction
p_ivtc
->
i_cadence_pos
=
0
;
/* Wild guess. The film frame reconstruction
will start in emergency mode, and this
will start in emergency mode, and this
will be filled in by the detector ASAP.*/
will be filled in by the detector ASAP.*/
/* I suppose video field dominance no longer flipflops. */
/* I suppose video field dominance no longer flipflops. */
p_ivtc
->
i_tfd
=
p_next
->
b_top_field_first
;
p_ivtc
->
i_tfd
=
!
p_next
->
b_top_field_first
;
/* tff <=> TFD == 0 */
}
}
}
}
...
@@ -4467,8 +4472,8 @@ static void IVTCCadenceAnalyze( filter_t *p_filter )
...
@@ -4467,8 +4472,8 @@ static void IVTCCadenceAnalyze( filter_t *p_filter )
}
}
/* See if the sequence is valid. The cadence positions must be
/* See if the sequence is valid. The cadence positions must be
successive mod 5.
We can't say anything about TFF/BFF yet,
successive mod 5. We can't say anything about TFF/BFF yet,
because the progressive-looking position "dea"
may be there.
because the progressive-looking position "dea" may be there.
If the sequence otherwise looks valid, we handle that last
If the sequence otherwise looks valid, we handle that last
by voting.
by voting.
...
@@ -4559,7 +4564,11 @@ static void IVTCCadenceAnalyze( filter_t *p_filter )
...
@@ -4559,7 +4564,11 @@ static void IVTCCadenceAnalyze( filter_t *p_filter )
/* If we still think the signal is progressive... */
/* If we still think the signal is progressive... */
if
(
b_really_all_progressive
)
if
(
b_really_all_progressive
)
{
{
/* ...exit film mode immediately. */
/* ...exit film mode immediately. This does not break
soft TC handling, because for soft TC at least one
of the frames will not qualify (due to i_nb_fields == 3),
and in that case this analysis will not run.
*/
if
(
p_ivtc
->
i_mode
==
IVTC_MODE_TELECINED_NTSC_HARD
)
if
(
p_ivtc
->
i_mode
==
IVTC_MODE_TELECINED_NTSC_HARD
)
msg_Dbg
(
p_filter
,
"IVTC: 3:2 pulldown: progressive "
\
msg_Dbg
(
p_filter
,
"IVTC: 3:2 pulldown: progressive "
\
"signal detected."
);
"signal detected."
);
...
@@ -4570,9 +4579,14 @@ static void IVTCCadenceAnalyze( filter_t *p_filter )
...
@@ -4570,9 +4579,14 @@ static void IVTCCadenceAnalyze( filter_t *p_filter )
}
}
/* Final missing "else": no valid NTSC telecine sequence detected.
/* Final missing "else": no valid NTSC telecine sequence detected.
Either there is no telecine, or the detector - although it produced
Either there is no telecine, or the detector - although it produced
results - had trouble finding it. In this case we do nothing,
results - had trouble finding it. In this case we do nothing,
as it's not a good idea to act on unreliable data.
as it's not a good idea to act on unreliable data.
Note that if we are already in IVTC_MODE_TELECINED_NTSC_HARD, this
case means that we have lost the lock-on, but are still (probably)
in a hard-telecined stream. This will start the emergency mode
for film frame reconstruction. See IVTCOutputOrDropFrame().
*/
*/
}
}
}
}
...
@@ -4616,19 +4630,21 @@ static bool IVTCOutputOrDropFrame( filter_t *p_filter, picture_t *p_dst )
...
@@ -4616,19 +4630,21 @@ static bool IVTCOutputOrDropFrame( filter_t *p_filter, picture_t *p_dst )
since been lost, and we have not seen a progressive signal after that.
since been lost, and we have not seen a progressive signal after that.
The latter case usually results from bad cuts, which interrupt
The latter case usually results from bad cuts, which interrupt
the cadence.
the cadence.
Lock-on state is given by p_ivtc->b_sequence_valid.
*/
*/
int
i_result_score
=
-
1
;
int
i_result_score
=
-
1
;
int
op
;
int
op
;
if
(
p_ivtc
->
i_mode
==
IVTC_MODE_TELECINED_NTSC_HARD
)
if
(
p_ivtc
->
i_mode
==
IVTC_MODE_TELECINED_NTSC_HARD
)
{
{
assert
(
p_ivtc
->
i_cadence_pos
!=
CADENCE_POS_INVALID
);
assert
(
p_ivtc
->
i_tfd
!=
TFD_INVALID
);
/* Decide what to do. The operation table is only enabled
/* Decide what to do. The operation table is only enabled
if the cadence seems reliable. Otherwise we use a backup strategy.
if the cadence seems reliable. Otherwise we use a backup strategy.
*/
*/
if
(
p_ivtc
->
b_sequence_valid
)
if
(
p_ivtc
->
b_sequence_valid
)
{
{
assert
(
p_ivtc
->
i_cadence_pos
!=
CADENCE_POS_INVALID
);
assert
(
p_ivtc
->
i_tfd
!=
TFD_INVALID
);
/* Pick correct operation from the operation table. */
/* Pick correct operation from the operation table. */
op
=
pi_reconstruction_ops
[
p_ivtc
->
i_tfd
][
p_ivtc
->
i_cadence_pos
];
op
=
pi_reconstruction_ops
[
p_ivtc
->
i_tfd
][
p_ivtc
->
i_cadence_pos
];
...
@@ -4640,8 +4656,7 @@ static bool IVTCOutputOrDropFrame( filter_t *p_filter, picture_t *p_dst )
...
@@ -4640,8 +4656,7 @@ static bool IVTCOutputOrDropFrame( filter_t *p_filter, picture_t *p_dst )
/* Drop frame. We're done. */
/* Drop frame. We're done. */
return
false
;
return
false
;
}
}
/* Frame not dropped */
else
else
if
(
p_ivtc
->
b_sequence_valid
)
{
{
if
(
op
==
IVTC_OP_COPY_N
)
if
(
op
==
IVTC_OP_COPY_N
)
i_result_score
=
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBN
];
i_result_score
=
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBN
];
...
@@ -4737,14 +4752,9 @@ static bool IVTCOutputOrDropFrame( filter_t *p_filter, picture_t *p_dst )
...
@@ -4737,14 +4752,9 @@ static bool IVTCOutputOrDropFrame( filter_t *p_filter, picture_t *p_dst )
}
}
}
}
/* Note that we get to this point only if we didn't drop the frame.
/* Mangle timestamps when locked on.
Mangle the presentation timestamp to convert 29.97 -> 23.976 fps.
*/
int
i_timestamp_delta
=
pi_timestamp_deltas
[
p_ivtc
->
i_cadence_pos
];
if
(
p_ivtc
->
b_sequence_valid
)
assert
(
i_timestamp_delta
>=
0
);
/*
"Current" is the frame that is being extracted now. Use its original
"Current" is the frame that is being extracted now. Use its original
timestamp as the base.
timestamp as the base.
Note that this way there will be no extra delay compared to the
Note that this way there will be no extra delay compared to the
...
@@ -4752,6 +4762,12 @@ static bool IVTCOutputOrDropFrame( filter_t *p_filter, picture_t *p_dst )
...
@@ -4752,6 +4762,12 @@ static bool IVTCOutputOrDropFrame( filter_t *p_filter, picture_t *p_dst )
*/
*/
if
(
p_ivtc
->
b_sequence_valid
)
if
(
p_ivtc
->
b_sequence_valid
)
{
{
/* Convert 29.97 -> 23.976 fps. We get to this point only if we
didn't drop the frame, so we always get a valid delta.
*/
int
i_timestamp_delta
=
pi_timestamp_deltas
[
p_ivtc
->
i_cadence_pos
];
assert
(
i_timestamp_delta
>=
0
);
/* FIXME: use field length as measured by Deinterlace()? */
/* FIXME: use field length as measured by Deinterlace()? */
t_final
=
p_curr
->
date
t_final
=
p_curr
->
date
+
(
p_next
->
date
-
p_curr
->
date
)
*
i_timestamp_delta
/
4
;
+
(
p_next
->
date
-
p_curr
->
date
)
*
i_timestamp_delta
/
4
;
...
@@ -4787,12 +4803,12 @@ static bool IVTCOutputOrDropFrame( filter_t *p_filter, picture_t *p_dst )
...
@@ -4787,12 +4803,12 @@ static bool IVTCOutputOrDropFrame( filter_t *p_filter, picture_t *p_dst )
/* Timestamp mangling for soft telecine: bump "threes" forward by
/* Timestamp mangling for soft telecine: bump "threes" forward by
0.5 field durations. This is more forgiving for the renderer
0.5 field durations. This is more forgiving for the renderer
than bumping the "twos" back (which would
require to render
than bumping the "twos" back (which would require to render
them sooner),
them sooner),
*/
*/
if
(
p_curr
->
i_nb_fields
==
3
)
if
(
p_curr
->
i_nb_fields
==
3
)
{
{
/* Approximate field duration from the PTS difference.
*/
/* Approximate field duration from the PTS difference. */
/* FIXME: use field length as measured by Deinterlace()? */
/* FIXME: use field length as measured by Deinterlace()? */
mtime_t
i_half_field_dur
=
(
(
p_next
->
date
-
p_curr
->
date
)
/
3
)
/
2
;
mtime_t
i_half_field_dur
=
(
(
p_next
->
date
-
p_curr
->
date
)
/
3
)
/
2
;
t_final
=
p_curr
->
date
+
i_half_field_dur
;
t_final
=
p_curr
->
date
+
i_half_field_dur
;
...
@@ -4931,31 +4947,31 @@ static int RenderIVTC( filter_t *p_filter, picture_t *p_dst, picture_t *p_src )
...
@@ -4931,31 +4947,31 @@ static int RenderIVTC( filter_t *p_filter, picture_t *p_dst, picture_t *p_src )
the only remaining possibility is that: */
the only remaining possibility is that: */
assert
(
p_curr
&&
!
p_prev
);
assert
(
p_curr
&&
!
p_prev
);
/* We need three frames for the detector to work, so we drop this one.
/* We need three frames for the cadence detector to work, so we just
We will only do some initialization for the detector here. */
do some init for the detector and pass the frame through.
Passthrough for second frame, too, works better than drop
for some still-image DVD menus.
Now that we have two frames, we can run a full IVTCLowLevelDetect().
The interlace scores from here will become TCBC, TCBP and TPBC
when the filter starts. The score for the current TCBC has already
been computed at the first frame, and slid into place at the start
of this frame (by IVTCFrameInit()).
*/
IVTCLowLevelDetect
(
p_filter
);
/* These scores will become TCBC, TCBP and TPBC when the filter starts.
/* Note that the sliding mechanism for output scores only starts
The score for the current TCBC has already been computed at the
when the actual filter does.
first frame, and slid into place at the start of this frame
(by IVTCFrameInit()).
*/
*/
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBN
]
=
CalculateInterlaceScore
(
p_next
,
p_next
);
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBC
]
=
CalculateInterlaceScore
(
p_next
,
p_curr
);
p_ivtc
->
pi_scores
[
FIELD_PAIR_TCBN
]
=
CalculateInterlaceScore
(
p_curr
,
p_next
);
/* TNBN is a wild guess, but doesn't really matter */
p_ivtc
->
pi_final_scores
[
1
]
=
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBN
];
p_ivtc
->
pi_final_scores
[
1
]
=
p_ivtc
->
pi_scores
[
FIELD_PAIR_TNBN
];
/* At the next frame, the filter starts. The next frame will get
/* At the next frame, the filter starts. The next frame will get
a custom timestamp. */
a custom timestamp. */
p_sys
->
i_frame_offset
=
CUSTOM_PTS
;
p_sys
->
i_frame_offset
=
CUSTOM_PTS
;
/* Not really an error. This is expected, but we must
picture_Copy
(
p_dst
,
p_next
);
signal the caller not to expect an output frame. */
return
VLC_SUCCESS
;
return
VLC_EGENERIC
;
}
}
}
}
...
@@ -4994,10 +5010,9 @@ static void IVTCClearState( filter_t *p_filter )
...
@@ -4994,10 +5010,9 @@ static void IVTCClearState( filter_t *p_filter )
p_ivtc
->
pi_v_raw
[
i
]
=
VEKTOR_CADENCE_POS_ALL
;
p_ivtc
->
pi_v_raw
[
i
]
=
VEKTOR_CADENCE_POS_ALL
;
/* the most neutral result considering the "vektor" algorithm */
p_ivtc
->
pi_top_rep
[
i
]
=
0
;
p_ivtc
->
pi_top_rep
[
i
]
=
1
;
p_ivtc
->
pi_bot_rep
[
i
]
=
0
;
p_ivtc
->
pi_bot_rep
[
i
]
=
1
;
p_ivtc
->
pi_motion
[
i
]
=
-
1
;
p_ivtc
->
pi_motion
[
i
]
=
-
1
;
p_ivtc
->
pb_all_progressives
[
i
]
=
false
;
p_ivtc
->
pb_all_progressives
[
i
]
=
false
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment