Add tests with out-of-order CEA-608 input buffers

The `Cea608DecoderTest` added here fails if re-ordering is removed from
`CeaDecoder`.

The `Cea608ParserTest` is added with `@Ignore` because there's currently
no re-ordering support in this part of the subtitle handling pipeline
(partly because there's no concept of 'current playback time', meaning
it's hard to know **when** to re-order).

PiperOrigin-RevId: 595320205
This commit is contained in:
ibaker 2024-01-03 00:43:29 -08:00 committed by Copybara-Service
parent 7b450f0d0a
commit 258ffa68b5
2 changed files with 135 additions and 17 deletions

View File

@ -71,8 +71,52 @@ public class Cea608DecoderTest {
createPacket(0xFC, 'p', 'l'), createPacket(0xFC, 'p', 'l'),
createPacket(0xFC, 'e', 's')); createPacket(0xFC, 'e', 's'));
Subtitle firstSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample1)); Subtitle firstSubtitle =
Subtitle secondSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample2)); checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 123, sample1));
Subtitle secondSubtitle =
checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 456, sample2));
assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("test subtitle, spa");
assertThat(getOnlyCue(secondSubtitle).text.toString())
.isEqualTo("test subtitle, spans 2 samples");
}
@Test
public void paintOnEmitsSubtitlesImmediately_reordersOutOfOrderSamples() throws Exception {
Cea608Decoder decoder =
new Cea608Decoder(
new Cea608Parser(
MimeTypes.APPLICATION_CEA608,
/* accessibilityChannel= */ 1,
Cea608Parser.MIN_DATA_CHANNEL_TIMEOUT_MS));
byte[] sample1 =
Bytes.concat(
// 'paint on' control character
createPacket(0xFC, 0x14, 0x29),
createPacket(0xFC, 't', 'e'),
createPacket(0xFC, 's', 't'),
createPacket(0xFC, ' ', 's'),
createPacket(0xFC, 'u', 'b'),
createPacket(0xFC, 't', 'i'),
createPacket(0xFC, 't', 'l'),
createPacket(0xFC, 'e', ','),
createPacket(0xFC, ' ', 's'),
createPacket(0xFC, 'p', 'a'));
byte[] sample2 =
Bytes.concat(
createPacket(0xFC, 'n', 's'),
createPacket(0xFC, ' ', '2'),
createPacket(0xFC, ' ', 's'),
createPacket(0xFC, 'a', 'm'),
createPacket(0xFC, 'p', 'l'),
createPacket(0xFC, 'e', 's'));
queueSample(decoder, /* timeUs= */ 456, sample2);
queueSample(decoder, /* timeUs= */ 123, sample1);
Subtitle firstSubtitle =
checkNotNull(decodeToPositionAndCopyResult(decoder, /* positionUs= */ 123));
Subtitle secondSubtitle =
checkNotNull(decodeToPositionAndCopyResult(decoder, /* positionUs= */ 456));
assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("test subtitle, spa"); assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("test subtitle, spa");
assertThat(getOnlyCue(secondSubtitle).text.toString()) assertThat(getOnlyCue(secondSubtitle).text.toString())
@ -120,9 +164,12 @@ public class Cea608DecoderTest {
createPacket(0xFC, 'n', 'e'), createPacket(0xFC, 'n', 'e'),
createPacket(0xFC, 's', 0x0)); createPacket(0xFC, 's', 0x0));
Subtitle firstSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample1)); Subtitle firstSubtitle =
Subtitle secondSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample2)); checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 123, sample1));
Subtitle thirdSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample3)); Subtitle secondSubtitle =
checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 456, sample2));
Subtitle thirdSubtitle =
checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 789, sample3));
assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("test subtitle, spa"); assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("test subtitle, spa");
assertThat(getOnlyCue(secondSubtitle).text.toString()) assertThat(getOnlyCue(secondSubtitle).text.toString())
@ -163,8 +210,10 @@ public class Cea608DecoderTest {
createPacket(0xFC, 'e', 0x0), createPacket(0xFC, 'e', 0x0),
createPacket(0xFD, 0x0, 0x0)); createPacket(0xFD, 0x0, 0x0));
Subtitle firstSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample1)); Subtitle firstSubtitle =
Subtitle secondSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample2)); checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 123, sample1));
Subtitle secondSubtitle =
checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 456, sample2));
assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("test sub"); assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("test sub");
assertThat(getOnlyCue(secondSubtitle).text.toString()).isEqualTo("test subtitle"); assertThat(getOnlyCue(secondSubtitle).text.toString()).isEqualTo("test subtitle");
@ -246,11 +295,16 @@ public class Cea608DecoderTest {
createPacket(0xFC, 'e', 0x0), createPacket(0xFC, 'e', 0x0),
createPacket(0xFD, 0x0, 0x0)); createPacket(0xFD, 0x0, 0x0));
Subtitle firstSubtitle = /*checkNotNull(*/ decodeSampleAndCopyResult(decoder, sample1) /*)*/; Subtitle firstSubtitle = /*checkNotNull(*/
Subtitle secondSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample2)); decodeSampleAndCopyResult(decoder, 123, sample1) /*)*/;
Subtitle thirdSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample3)); Subtitle secondSubtitle =
Subtitle fourthSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample4)); checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 456, sample2));
Subtitle fifthSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample5)); Subtitle thirdSubtitle =
checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 789, sample3));
Subtitle fourthSubtitle =
checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 1234, sample4));
Subtitle fifthSubtitle =
checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 5678, sample5));
assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("te"); assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("te");
assertThat(getOnlyCue(secondSubtitle).text.toString()).isEqualTo("test"); assertThat(getOnlyCue(secondSubtitle).text.toString()).isEqualTo("test");
@ -284,7 +338,8 @@ public class Cea608DecoderTest {
createPacket(0xFC, 'X', 'X'), createPacket(0xFC, 'X', 'X'),
createPacket(0xFD, 0x0, 0x0)); createPacket(0xFD, 0x0, 0x0));
Subtitle firstSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample1)); Subtitle firstSubtitle =
checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 123, sample1));
assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("test"); assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("test");
} }
@ -315,7 +370,8 @@ public class Cea608DecoderTest {
createPacket(0xFC, 0x0, 0x0), createPacket(0xFC, 0x0, 0x0),
createPacket(0xFD, 'X', 'X')); createPacket(0xFD, 'X', 'X'));
Subtitle firstSubtitle = checkNotNull(decodeSampleAndCopyResult(decoder, sample1)); Subtitle firstSubtitle =
checkNotNull(decodeSampleAndCopyResult(decoder, /* timeUs= */ 123, sample1));
assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("test"); assertThat(getOnlyCue(firstSubtitle).text.toString()).isEqualTo("test");
} }
@ -336,17 +392,39 @@ public class Cea608DecoderTest {
} }
/** /**
* Queues {@code sample} to {@code decoder} and dequeues the result, then copies and returns it if * Queues {@code sample} to {@code decoder} with {@code timeUs}, sets {@link
* it's non-null. * Cea608Decoder#setPositionUs(long)} to {@code timeUs} and dequeues the result, then copies and
* returns it if it's non-null.
* *
* <p>Fails if {@link Cea608Decoder#dequeueInputBuffer()} returns {@code null}. * <p>Fails if {@link Cea608Decoder#dequeueInputBuffer()} returns {@code null}.
*/ */
@Nullable @Nullable
private static Subtitle decodeSampleAndCopyResult(Cea608Decoder decoder, byte[] sample) private static Subtitle decodeSampleAndCopyResult(
Cea608Decoder decoder, long timeUs, byte[] sample) throws SubtitleDecoderException {
queueSample(decoder, timeUs, sample);
return decodeToPositionAndCopyResult(decoder, timeUs);
}
/**
* Queues {@code sample} to {@code decoder}. Fails if {@link Cea608Decoder#dequeueInputBuffer()}
* returns {@code null}.
*/
private static void queueSample(Cea608Decoder decoder, long timeUs, byte[] sample)
throws SubtitleDecoderException { throws SubtitleDecoderException {
SubtitleInputBuffer inputBuffer = checkNotNull(decoder.dequeueInputBuffer()); SubtitleInputBuffer inputBuffer = checkNotNull(decoder.dequeueInputBuffer());
inputBuffer.data = ByteBuffer.wrap(sample); inputBuffer.data = ByteBuffer.wrap(sample);
inputBuffer.timeUs = timeUs;
decoder.queueInputBuffer(inputBuffer); decoder.queueInputBuffer(inputBuffer);
}
/**
* Sets {@link Cea608Decoder#setPositionUs(long)} then dequeues the result, and copies and returns
* it if it's non-null.
*/
@Nullable
private static Subtitle decodeToPositionAndCopyResult(Cea608Decoder decoder, long positionUs)
throws SubtitleDecoderException {
decoder.setPositionUs(positionUs);
@Nullable SubtitleOutputBuffer outputBuffer = decoder.dequeueOutputBuffer(); @Nullable SubtitleOutputBuffer outputBuffer = decoder.dequeueOutputBuffer();
if (outputBuffer == null) { if (outputBuffer == null) {
return null; return null;

View File

@ -30,6 +30,7 @@ import com.google.common.primitives.Bytes;
import com.google.common.primitives.UnsignedBytes; import com.google.common.primitives.UnsignedBytes;
import java.util.ArrayList; import java.util.ArrayList;
import java.util.List; import java.util.List;
import org.junit.Ignore;
import org.junit.Test; import org.junit.Test;
import org.junit.runner.RunWith; import org.junit.runner.RunWith;
@ -122,6 +123,45 @@ public class Cea608ParserTest {
.isEqualTo("test subtitle, spans 2 samples"); .isEqualTo("test subtitle, spans 2 samples");
} }
@Test
@Ignore("Out-of-order CEA-608 samples are not yet supported (internal b/317488646).")
public void paintOnEmitsSubtitlesImmediately_reordersOutOfOrderSamples() throws Exception {
Cea608Parser cea608Parser =
new Cea608Parser(
MimeTypes.APPLICATION_CEA608,
/* accessibilityChannel= */ 1,
Cea608Parser.MIN_DATA_CHANNEL_TIMEOUT_MS);
byte[] sample1 =
Bytes.concat(
// 'paint on' control character
createPacket(0xFC, 0x14, 0x29),
createPacket(0xFC, 't', 'e'),
createPacket(0xFC, 's', 't'),
createPacket(0xFC, ' ', 's'),
createPacket(0xFC, 'u', 'b'),
createPacket(0xFC, 't', 'i'),
createPacket(0xFC, 't', 'l'),
createPacket(0xFC, 'e', ','),
createPacket(0xFC, ' ', 's'),
createPacket(0xFC, 'p', 'a'));
byte[] sample2 =
Bytes.concat(
createPacket(0xFC, 'n', 's'),
createPacket(0xFC, ' ', '2'),
createPacket(0xFC, ' ', 's'),
createPacket(0xFC, 'a', 'm'),
createPacket(0xFC, 'p', 'l'),
createPacket(0xFC, 'e', 's'));
CuesWithTiming secondCues = checkNotNull(parseSample(cea608Parser, sample2));
CuesWithTiming firstCues = checkNotNull(parseSample(cea608Parser, sample1));
assertThat(Iterables.getOnlyElement(firstCues.cues).text.toString())
.isEqualTo("test subtitle, spa");
assertThat(Iterables.getOnlyElement(secondCues.cues).text.toString())
.isEqualTo("test subtitle, spans 2 samples");
}
@Test @Test
public void rollUpEmitsSubtitlesImmediately() throws Exception { public void rollUpEmitsSubtitlesImmediately() throws Exception {
Cea608Parser cea608Parser = Cea608Parser cea608Parser =