Skip to content

Commit 72da940

Browse files
committed
Move TS binary search to using DTS; fix PCR based duration; update and add dumps; add .gitattributes
1 parent 1554589 commit 72da940

File tree

75 files changed

+828
-770
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

75 files changed

+828
-770
lines changed

libraries/extractor/src/main/java/androidx/media3/extractor/ts/TsBinarySearchSeeker.java

+33-33
Original file line numberDiff line numberDiff line change
@@ -28,25 +28,25 @@
2828
/**
2929
* A seeker that supports seeking within TS stream using binary search.
3030
*
31-
* <p>This seeker uses the first and last PCR values within the stream, as well as the stream
32-
* duration to interpolate the PCR value of the seeking position. Then it performs binary search
33-
* within the stream to find a packets whose PCR value is within {@link #SEEK_TOLERANCE_US} from the
34-
* target PCR.
31+
* <p>This seeker uses the first and last DTS values within the stream, as well as the stream
32+
* duration to interpolate the DTS value of the seeking position. Then it performs binary search
33+
* within the stream to find a packets whose DTS value is within {@link #SEEK_TOLERANCE_US} from the
34+
* target DTS.
3535
*/
3636
/* package */ final class TsBinarySearchSeeker extends BinarySearchSeeker {
3737

3838
private static final long SEEK_TOLERANCE_US = 100_000;
3939
private static final int MINIMUM_SEARCH_RANGE_BYTES = 5 * TsExtractor.TS_PACKET_SIZE;
4040

4141
public TsBinarySearchSeeker(
42-
TimestampAdjuster pcrTimestampAdjuster,
42+
TimestampAdjuster timestampAdjuster,
4343
long streamDurationUs,
4444
long inputLength,
45-
int pcrPid,
45+
int selectedPid,
4646
int timestampSearchBytes) {
4747
super(
4848
new DefaultSeekTimestampConverter(),
49-
new TsPcrSeeker(pcrPid, pcrTimestampAdjuster, timestampSearchBytes),
49+
new TsTimestampSeeker(selectedPid, timestampAdjuster, timestampSearchBytes),
5050
streamDurationUs,
5151
/* floorTimePosition= */ 0,
5252
/* ceilingTimePosition= */ streamDurationUs + 1,
@@ -57,25 +57,25 @@ public TsBinarySearchSeeker(
5757
}
5858

5959
/**
60-
* A {@link TimestampSeeker} implementation that looks for a given PCR timestamp at a given
60+
* A {@link TimestampSeeker} implementation that looks for a given DTS timestamp at a given
6161
* position in a TS stream.
6262
*
63-
* <p>Given a PCR timestamp, and a position within a TS stream, this seeker will peek up to {@link
63+
* <p>Given a DTS timestamp, and a position within a TS stream, this seeker will peek up to {@link
6464
* #timestampSearchBytes} from that stream position, look for all packets with PID equal to
65-
* PCR_PID, and then compare the PCR timestamps (if available) of these packets to the target
65+
* SELECTED_PID, and then compare the DTS timestamps (if available) of these packets to the target
6666
* timestamp.
6767
*/
68-
private static final class TsPcrSeeker implements TimestampSeeker {
68+
private static final class TsTimestampSeeker implements TimestampSeeker {
6969

70-
private final TimestampAdjuster pcrTimestampAdjuster;
70+
private final TimestampAdjuster timestampAdjuster;
7171
private final ParsableByteArray packetBuffer;
72-
private final int pcrPid;
72+
private final int selectedPid;
7373
private final int timestampSearchBytes;
7474

75-
public TsPcrSeeker(
76-
int pcrPid, TimestampAdjuster pcrTimestampAdjuster, int timestampSearchBytes) {
77-
this.pcrPid = pcrPid;
78-
this.pcrTimestampAdjuster = pcrTimestampAdjuster;
75+
public TsTimestampSeeker(
76+
int selectedPid, TimestampAdjuster timestampAdjuster, int timestampSearchBytes) {
77+
this.selectedPid = selectedPid;
78+
this.timestampAdjuster = timestampAdjuster;
7979
this.timestampSearchBytes = timestampSearchBytes;
8080
packetBuffer = new ParsableByteArray();
8181
}
@@ -89,16 +89,16 @@ public TimestampSearchResult searchForTimestamp(ExtractorInput input, long targe
8989
packetBuffer.reset(bytesToSearch);
9090
input.peekFully(packetBuffer.getData(), /* offset= */ 0, bytesToSearch);
9191

92-
return searchForPcrValueInBuffer(packetBuffer, targetTimestamp, inputPosition);
92+
return searchForDtsValueInBuffer(packetBuffer, targetTimestamp, inputPosition);
9393
}
9494

95-
private TimestampSearchResult searchForPcrValueInBuffer(
96-
ParsableByteArray packetBuffer, long targetPcrTimeUs, long bufferStartOffset) {
95+
private TimestampSearchResult searchForDtsValueInBuffer(
96+
ParsableByteArray packetBuffer, long targetDtsTimeUs, long bufferStartOffset) {
9797
int limit = packetBuffer.limit();
9898

9999
long startOfLastPacketPosition = C.INDEX_UNSET;
100100
long endOfLastPacketPosition = C.INDEX_UNSET;
101-
long lastPcrTimeUsInRange = C.TIME_UNSET;
101+
long lastDtsTimeUsInRange = C.TIME_UNSET;
102102

103103
while (packetBuffer.bytesLeft() >= TsExtractor.TS_PACKET_SIZE) {
104104
int startOfPacket =
@@ -107,34 +107,34 @@ private TimestampSearchResult searchForPcrValueInBuffer(
107107
if (endOfPacket > limit) {
108108
break;
109109
}
110-
long pcrValue = TsUtil.readPcrFromPacket(packetBuffer, startOfPacket, pcrPid);
111-
if (pcrValue != C.TIME_UNSET) {
112-
long pcrTimeUs = pcrTimestampAdjuster.adjustTsTimestamp(pcrValue);
113-
if (pcrTimeUs > targetPcrTimeUs) {
114-
if (lastPcrTimeUsInRange == C.TIME_UNSET) {
115-
// First PCR timestamp is already over target.
116-
return TimestampSearchResult.overestimatedResult(pcrTimeUs, bufferStartOffset);
110+
long dtsValue = TsUtil.readDtsFromPacket(packetBuffer, startOfPacket, selectedPid);
111+
if (dtsValue != C.TIME_UNSET) {
112+
long dtsTimeUs = timestampAdjuster.adjustTsTimestamp(dtsValue);
113+
if (dtsTimeUs > targetDtsTimeUs) {
114+
if (lastDtsTimeUsInRange == C.TIME_UNSET) {
115+
// First DTS timestamp is already over target.
116+
return TimestampSearchResult.overestimatedResult(dtsTimeUs, bufferStartOffset);
117117
} else {
118-
// Last PCR timestamp < target timestamp < this timestamp.
118+
// Last DTS timestamp < target timestamp < this timestamp.
119119
return TimestampSearchResult.targetFoundResult(
120120
bufferStartOffset + startOfLastPacketPosition);
121121
}
122-
} else if (pcrTimeUs + SEEK_TOLERANCE_US > targetPcrTimeUs) {
122+
} else if (dtsTimeUs + SEEK_TOLERANCE_US > targetDtsTimeUs) {
123123
long startOfPacketInStream = bufferStartOffset + startOfPacket;
124124
return TimestampSearchResult.targetFoundResult(startOfPacketInStream);
125125
}
126126

127-
lastPcrTimeUsInRange = pcrTimeUs;
127+
lastDtsTimeUsInRange = dtsTimeUs;
128128
startOfLastPacketPosition = startOfPacket;
129129
}
130130
packetBuffer.setPosition(endOfPacket);
131131
endOfLastPacketPosition = endOfPacket;
132132
}
133133

134-
if (lastPcrTimeUsInRange != C.TIME_UNSET) {
134+
if (lastDtsTimeUsInRange != C.TIME_UNSET) {
135135
long endOfLastPacketPositionInStream = bufferStartOffset + endOfLastPacketPosition;
136136
return TimestampSearchResult.underestimatedResult(
137-
lastPcrTimeUsInRange, endOfLastPacketPositionInStream);
137+
lastDtsTimeUsInRange, endOfLastPacketPositionInStream);
138138
} else {
139139
return TimestampSearchResult.NO_TIMESTAMP_IN_RANGE_RESULT;
140140
}

libraries/extractor/src/main/java/androidx/media3/extractor/ts/TsDurationReader.java

+19-15
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,7 @@
1616
package androidx.media3.extractor.ts;
1717

1818
import static java.lang.Math.min;
19+
import static java.lang.Math.max;
1920

2021
import androidx.media3.common.C;
2122
import androidx.media3.common.util.ParsableByteArray;
@@ -61,6 +62,7 @@
6162
private long secondDtsValue;
6263
private long lastDtsValue;
6364

65+
private int firstPcrPosition;
6466
private int firstDtsPosition;
6567

6668
private long durationUs;
@@ -73,6 +75,7 @@
7375
firstDtsValue = C.TIME_UNSET;
7476
secondDtsValue = C.TIME_UNSET;
7577
lastDtsValue = C.TIME_UNSET;
78+
firstPcrPosition = C.INDEX_UNSET;
7679
firstDtsPosition = C.INDEX_UNSET;
7780
durationUs = C.TIME_UNSET;
7881
packetBuffer = new ParsableByteArray();
@@ -126,17 +129,17 @@ public boolean isDurationReadFinished() {
126129
isDurationReadFromPcr = true;
127130
return Extractor.RESULT_CONTINUE;
128131
}
129-
if (!isLastPcrValueRead) {
130-
return readLastPcrValue(input, seekPositionHolder, pcrPid);
132+
if (!isFirstPcrValueRead) {
133+
return readFirstPcrValue(input, seekPositionHolder, pcrPid);
131134
}
132-
if (lastPcrValue == C.TIME_UNSET) {
135+
if (firstPcrValue == C.TIME_UNSET) {
133136
isDurationReadFromPcr = true;
134137
return Extractor.RESULT_CONTINUE;
135138
}
136-
if (!isFirstPcrValueRead) {
137-
return readFirstPcrValue(input, seekPositionHolder, pcrPid);
139+
if (!isLastPcrValueRead) {
140+
return readLastPcrValue(input, seekPositionHolder, pcrPid);
138141
}
139-
if (firstPcrValue == C.TIME_UNSET) {
142+
if (lastPcrValue == C.TIME_UNSET) {
140143
isDurationReadFromPcr = true;
141144
return Extractor.RESULT_CONTINUE;
142145
}
@@ -171,13 +174,6 @@ public boolean isDurationReadFinished() {
171174
isDurationReadFromDts = true;
172175
return Extractor.RESULT_CONTINUE;
173176
}
174-
if (!isLastDtsValueRead) {
175-
return readLastDtsValue(input, seekPositionHolder, pesPid);
176-
}
177-
if (lastDtsValue == C.TIME_UNSET) {
178-
isDurationReadFromDts = true;
179-
return Extractor.RESULT_CONTINUE;
180-
}
181177
if (!isFirstDtsValueRead) {
182178
return readFirstDtsValue(input, seekPositionHolder, pesPid);
183179
}
@@ -192,6 +188,13 @@ public boolean isDurationReadFinished() {
192188
isDurationReadFromDts = true;
193189
return Extractor.RESULT_CONTINUE;
194190
}
191+
if (!isLastDtsValueRead) {
192+
return readLastDtsValue(input, seekPositionHolder, pesPid);
193+
}
194+
if (lastDtsValue == C.TIME_UNSET) {
195+
isDurationReadFromDts = true;
196+
return Extractor.RESULT_CONTINUE;
197+
}
195198

196199
long firstDtsPositionUs = timestampAdjuster.adjustTsTimestamp(firstDtsValue);
197200
long secondDtsPositionUs = timestampAdjuster.adjustTsTimestamp(secondDtsValue);
@@ -256,6 +259,7 @@ private long readFirstPcrValueFromBuffer(ParsableByteArray packetBuffer, int pcr
256259
}
257260
long pcrValue = TsUtil.readPcrFromPacket(packetBuffer, searchPosition, pcrPid);
258261
if (pcrValue != C.TIME_UNSET) {
262+
firstPcrPosition = packetBuffer.getPosition();
259263
return pcrValue;
260264
}
261265
}
@@ -265,7 +269,7 @@ private long readFirstPcrValueFromBuffer(ParsableByteArray packetBuffer, int pcr
265269
private int readLastPcrValue(ExtractorInput input, PositionHolder seekPositionHolder, int pcrPid)
266270
throws IOException {
267271
long inputLength = input.getLength();
268-
int bytesToSearch = (int) min(timestampSearchBytes, inputLength);
272+
int bytesToSearch = (int) min(timestampSearchBytes, inputLength - firstPcrPosition);
269273
long searchStartPosition = inputLength - bytesToSearch;
270274
if (input.getPosition() != searchStartPosition) {
271275
seekPositionHolder.position = searchStartPosition;
@@ -358,7 +362,7 @@ private int readSecondDtsValue(ExtractorInput input, PositionHolder seekPosition
358362
private int readLastDtsValue(ExtractorInput input, PositionHolder seekPositionHolder, int pesPid)
359363
throws IOException {
360364
long inputLength = input.getLength();
361-
int bytesToSearch = (int) min(timestampSearchBytes, inputLength);
365+
int bytesToSearch = (int) min(timestampSearchBytes, inputLength - firstDtsPosition);
362366
long searchStartPosition = inputLength - bytesToSearch;
363367
if (input.getPosition() != searchStartPosition) {
364368
seekPositionHolder.position = searchStartPosition;

libraries/test_data/src/test/assets/extractordumps/ts/sample_ac3.ts.0.dump

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,10 @@
11
seekMap:
22
isSeekable = true
3-
duration = 252977
3+
duration = 288000
44
getPosition(0) = [[timeUs=0, position=0]]
55
getPosition(1) = [[timeUs=1, position=0]]
6-
getPosition(126488) = [[timeUs=126488, position=9099]]
7-
getPosition(252977) = [[timeUs=252977, position=18386]]
6+
getPosition(144000) = [[timeUs=144000, position=9099]]
7+
getPosition(288000) = [[timeUs=288000, position=18386]]
88
numberOfTracks = 1
99
track 1900:
1010
total output bytes = 13281

libraries/test_data/src/test/assets/extractordumps/ts/sample_ac3.ts.1.dump

+9-13
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,14 @@
11
seekMap:
22
isSeekable = true
3-
duration = 252977
3+
duration = 288000
44
getPosition(0) = [[timeUs=0, position=0]]
55
getPosition(1) = [[timeUs=1, position=0]]
6-
getPosition(126488) = [[timeUs=126488, position=9099]]
7-
getPosition(252977) = [[timeUs=252977, position=18386]]
6+
getPosition(144000) = [[timeUs=144000, position=9099]]
7+
getPosition(288000) = [[timeUs=288000, position=18386]]
88
numberOfTracks = 1
99
track 1900:
10-
total output bytes = 10209
11-
sample count = 6
10+
total output bytes = 8673
11+
sample count = 5
1212
format 0:
1313
averageBitrate = 384000
1414
peakBitrate = 384000
@@ -17,26 +17,22 @@ track 1900:
1717
channelCount = 6
1818
sampleRate = 48000
1919
sample 0:
20-
time = 96000
21-
flags = 1
22-
data = length 1536, hash 5D09685
23-
sample 1:
2420
time = 128000
2521
flags = 1
2622
data = length 1536, hash A9A24E44
27-
sample 2:
23+
sample 1:
2824
time = 160000
2925
flags = 1
3026
data = length 1536, hash 6F856273
31-
sample 3:
27+
sample 2:
3228
time = 192000
3329
flags = 1
3430
data = length 1536, hash B1737D3C
35-
sample 4:
31+
sample 3:
3632
time = 224000
3733
flags = 1
3834
data = length 1536, hash 98FDEB9D
39-
sample 5:
35+
sample 4:
4036
time = 256000
4137
flags = 1
4238
data = length 1536, hash 99B9B943

libraries/test_data/src/test/assets/extractordumps/ts/sample_ac3.ts.2.dump

+6-14
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,14 @@
11
seekMap:
22
isSeekable = true
3-
duration = 252977
3+
duration = 288000
44
getPosition(0) = [[timeUs=0, position=0]]
55
getPosition(1) = [[timeUs=1, position=0]]
6-
getPosition(126488) = [[timeUs=126488, position=9099]]
7-
getPosition(252977) = [[timeUs=252977, position=18386]]
6+
getPosition(144000) = [[timeUs=144000, position=9099]]
7+
getPosition(288000) = [[timeUs=288000, position=18386]]
88
numberOfTracks = 1
99
track 1900:
10-
total output bytes = 7137
11-
sample count = 4
10+
total output bytes = 4065
11+
sample count = 2
1212
format 0:
1313
averageBitrate = 384000
1414
peakBitrate = 384000
@@ -17,18 +17,10 @@ track 1900:
1717
channelCount = 6
1818
sampleRate = 48000
1919
sample 0:
20-
time = 160000
21-
flags = 1
22-
data = length 1536, hash 6F856273
23-
sample 1:
24-
time = 192000
25-
flags = 1
26-
data = length 1536, hash B1737D3C
27-
sample 2:
2820
time = 224000
2921
flags = 1
3022
data = length 1536, hash 98FDEB9D
31-
sample 3:
23+
sample 1:
3224
time = 256000
3325
flags = 1
3426
data = length 1536, hash 99B9B943

libraries/test_data/src/test/assets/extractordumps/ts/sample_ac3.ts.3.dump

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,10 @@
11
seekMap:
22
isSeekable = true
3-
duration = 252977
3+
duration = 288000
44
getPosition(0) = [[timeUs=0, position=0]]
55
getPosition(1) = [[timeUs=1, position=0]]
6-
getPosition(126488) = [[timeUs=126488, position=9099]]
7-
getPosition(252977) = [[timeUs=252977, position=18386]]
6+
getPosition(144000) = [[timeUs=144000, position=9099]]
7+
getPosition(288000) = [[timeUs=288000, position=18386]]
88
numberOfTracks = 1
99
track 1900:
1010
total output bytes = 0

libraries/test_data/src/test/assets/extractordumps/ts/sample_ac4.ts.0.dump

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,10 @@
11
seekMap:
22
isSeekable = true
3-
duration = 796888
3+
duration = 760000
44
getPosition(0) = [[timeUs=0, position=0]]
55
getPosition(1) = [[timeUs=1, position=0]]
6-
getPosition(398444) = [[timeUs=398444, position=13385]]
7-
getPosition(796888) = [[timeUs=796888, position=26959]]
6+
getPosition(380000) = [[timeUs=380000, position=13385]]
7+
getPosition(760000) = [[timeUs=760000, position=26959]]
88
numberOfTracks = 1
99
track 1900:
1010
total output bytes = 7594

0 commit comments

Comments
 (0)