diff --git a/lib/media/media_source_capabilities.js b/lib/media/media_source_capabilities.js index 8c0e0f2f64e..fa6c90cc587 100644 --- a/lib/media/media_source_capabilities.js +++ b/lib/media/media_source_capabilities.js @@ -27,6 +27,16 @@ shaka.media.Capabilities = class { supportMap.set(type, currentSupport); return currentSupport; } + + /** + * Determine support for SourceBuffer.changeType + * @return {boolean} + */ + static isChangeTypeSupported() { + return !!window.SourceBuffer && + // eslint-disable-next-line no-restricted-syntax + !!SourceBuffer.prototype && !!SourceBuffer.prototype.changeType; + } }; /** diff --git a/lib/media/media_source_engine.js b/lib/media/media_source_engine.js index 97d8087dd27..4837476084d 100644 --- a/lib/media/media_source_engine.js +++ b/lib/media/media_source_engine.js @@ -118,6 +118,9 @@ shaka.media.MediaSourceEngine = class { /** @private {MediaSource} */ this.mediaSource_ = this.createMediaSource(this.mediaSourceOpen_); + /** @private {boolean} */ + this.reloadingMediaSource_ = false; + /** @type {!shaka.util.Destroyer} */ this.destroyer_ = new shaka.util.Destroyer(() => this.doDestroy_()); @@ -372,8 +375,6 @@ shaka.media.MediaSourceEngine = class { async init(streamsByType, sequenceMode=false, manifestType=shaka.media.ManifestParser.UNKNOWN, ignoreManifestTimestampsInSegmentsMode=false) { - const ContentType = shaka.util.ManifestParserUtils.ContentType; - await this.mediaSourceOpen_; this.sequenceMode_ = sequenceMode; @@ -383,65 +384,78 @@ shaka.media.MediaSourceEngine = class { for (const contentType of streamsByType.keys()) { const stream = streamsByType.get(contentType); - goog.asserts.assert( - shaka.media.MediaSourceEngine.isStreamSupported(stream), - 'Type negotiation should happen before MediaSourceEngine.init!'); - - let mimeType = shaka.util.MimeUtils.getFullType( - stream.mimeType, stream.codecs); - if (contentType == ContentType.TEXT) { - this.reinitText(mimeType, sequenceMode); - } else { - let needTransmux = this.config_.forceTransmux; - if (!shaka.media.Capabilities.isTypeSupported(mimeType) || - (!sequenceMode && - shaka.util.MimeUtils.RAW_FORMATS.includes(mimeType))) { - needTransmux = true; - } - const mimeTypeWithAllCodecs = - shaka.util.MimeUtils.getFullTypeWithAllCodecs( - stream.mimeType, stream.codecs); - const TransmuxerEngine = shaka.transmuxer.TransmuxerEngine; - if (needTransmux && - TransmuxerEngine.isSupported(mimeTypeWithAllCodecs, contentType)) { - const transmuxerPlugin = - TransmuxerEngine.findTransmuxer(mimeTypeWithAllCodecs); - if (transmuxerPlugin) { - const transmuxer = transmuxerPlugin(); - this.transmuxers_[contentType] = transmuxer; - mimeType = - transmuxer.convertCodecs(contentType, mimeTypeWithAllCodecs); - } - } - const type = mimeType + this.config_.sourceBufferExtraFeatures; + this.initSourceBuffer_(contentType, stream); + this.queues_[contentType] = []; + } + } - this.destroyer_.ensureNotDestroyed(); + /** + * Initialize a specific SourceBuffer. + * + * @param {shaka.util.ManifestParserUtils.ContentType} contentType + * @param {shaka.extern.Stream} stream + * @private + */ + initSourceBuffer_(contentType, stream) { + const ContentType = shaka.util.ManifestParserUtils.ContentType; - let sourceBuffer; + goog.asserts.assert( + shaka.media.MediaSourceEngine.isStreamSupported(stream), + 'Type negotiation should happen before MediaSourceEngine.init!'); - try { - sourceBuffer = this.mediaSource_.addSourceBuffer(type); - } catch (exception) { - throw new shaka.util.Error( - shaka.util.Error.Severity.CRITICAL, - shaka.util.Error.Category.MEDIA, - shaka.util.Error.Code.MEDIA_SOURCE_OPERATION_THREW, - exception, - 'The mediaSource_ status was' + this.mediaSource_.readyState + - 'expected \'open\''); + let mimeType = shaka.util.MimeUtils.getFullType( + stream.mimeType, stream.codecs); + if (contentType == ContentType.TEXT) { + this.reinitText(mimeType, this.sequenceMode_); + } else { + let needTransmux = this.config_.forceTransmux; + if (!shaka.media.Capabilities.isTypeSupported(mimeType) || + (!this.sequenceMode_ && + shaka.util.MimeUtils.RAW_FORMATS.includes(mimeType))) { + needTransmux = true; + } + const mimeTypeWithAllCodecs = + shaka.util.MimeUtils.getFullTypeWithAllCodecs( + stream.mimeType, stream.codecs); + const TransmuxerEngine = shaka.transmuxer.TransmuxerEngine; + if (needTransmux && + TransmuxerEngine.isSupported(mimeTypeWithAllCodecs, contentType)) { + const transmuxerPlugin = + TransmuxerEngine.findTransmuxer(mimeTypeWithAllCodecs); + if (transmuxerPlugin) { + const transmuxer = transmuxerPlugin(); + this.transmuxers_[contentType] = transmuxer; + mimeType = + transmuxer.convertCodecs(contentType, mimeTypeWithAllCodecs); } + } + const type = mimeType + this.config_.sourceBufferExtraFeatures; + + this.destroyer_.ensureNotDestroyed(); - this.eventManager_.listen( - sourceBuffer, 'error', - () => this.onError_(contentType)); - this.eventManager_.listen( - sourceBuffer, 'updateend', - () => this.onUpdateEnd_(contentType)); - this.sourceBuffers_[contentType] = sourceBuffer; - this.sourceBufferTypes_[contentType] = mimeType; - this.queues_[contentType] = []; - this.expectedEncryption_[contentType] = !!stream.drmInfos.length; + let sourceBuffer; + + try { + sourceBuffer = this.mediaSource_.addSourceBuffer(type); + } catch (exception) { + throw new shaka.util.Error( + shaka.util.Error.Severity.CRITICAL, + shaka.util.Error.Category.MEDIA, + shaka.util.Error.Code.MEDIA_SOURCE_OPERATION_THREW, + exception, + 'The mediaSource_ status was' + this.mediaSource_.readyState + + 'expected \'open\''); } + + this.eventManager_.listen( + sourceBuffer, 'error', + () => this.onError_(contentType)); + this.eventManager_.listen( + sourceBuffer, 'updateend', + () => this.onUpdateEnd_(contentType)); + this.sourceBuffers_[contentType] = sourceBuffer; + this.sourceBufferTypes_[contentType] = mimeType; + this.expectedEncryption_[contentType] = !!stream.drmInfos.length; } } @@ -473,6 +487,9 @@ shaka.media.MediaSourceEngine = class { * object has been destroyed. */ ended() { + if (this.reloadingMediaSource_) { + return false; + } return this.mediaSource_ ? this.mediaSource_.readyState == 'ended' : true; } @@ -483,6 +500,9 @@ shaka.media.MediaSourceEngine = class { * @return {?number} The timestamp in seconds, or null if nothing is buffered. */ bufferStart(contentType) { + if (this.reloadingMediaSource_) { + return null; + } const ContentType = shaka.util.ManifestParserUtils.ContentType; if (contentType == ContentType.TEXT) { return this.textEngine_.bufferStart(); @@ -498,6 +518,9 @@ shaka.media.MediaSourceEngine = class { * @return {?number} The timestamp in seconds, or null if nothing is buffered. */ bufferEnd(contentType) { + if (this.reloadingMediaSource_) { + return null; + } const ContentType = shaka.util.ManifestParserUtils.ContentType; if (contentType == ContentType.TEXT) { return this.textEngine_.bufferEnd(); @@ -515,6 +538,9 @@ shaka.media.MediaSourceEngine = class { * @return {boolean} */ isBuffered(contentType, time) { + if (this.reloadingMediaSource_) { + return false; + } const ContentType = shaka.util.ManifestParserUtils.ContentType; if (contentType == ContentType.TEXT) { return this.textEngine_.isBuffered(time); @@ -533,6 +559,9 @@ shaka.media.MediaSourceEngine = class { * @return {number} The amount of time buffered ahead in seconds. */ bufferedAheadOf(contentType, time) { + if (this.reloadingMediaSource_) { + return 0; + } const ContentType = shaka.util.ManifestParserUtils.ContentType; if (contentType == ContentType.TEXT) { return this.textEngine_.bufferedAheadOf(time); @@ -551,11 +580,12 @@ shaka.media.MediaSourceEngine = class { const TimeRangesUtils = shaka.media.TimeRangesUtils; const info = { - total: TimeRangesUtils.getBufferedInfo(this.video_.buffered), - audio: TimeRangesUtils.getBufferedInfo( - this.getBuffered_(ContentType.AUDIO)), - video: TimeRangesUtils.getBufferedInfo( - this.getBuffered_(ContentType.VIDEO)), + total: this.reloadingMediaSource_ ? [] : + TimeRangesUtils.getBufferedInfo(this.video_.buffered), + audio: this.reloadingMediaSource_ ? [] : + TimeRangesUtils.getBufferedInfo(this.getBuffered_(ContentType.AUDIO)), + video: this.reloadingMediaSource_ ? [] : + TimeRangesUtils.getBufferedInfo(this.getBuffered_(ContentType.VIDEO)), text: [], }; @@ -1002,11 +1032,17 @@ shaka.media.MediaSourceEngine = class { * value will be dropped. * @param {boolean} sequenceMode If true, the timestampOffset will not be * applied in this step. + * @param {shaka.extern.Stream} stream The current stream. + * @param {!Map.} streamsByType + * A map of content types to streams. All streams must be supported + * according to MediaSourceEngine.isStreamSupported. + * * @return {!Promise} */ async setStreamProperties( contentType, timestampOffset, appendWindowStart, appendWindowEnd, - sequenceMode) { + sequenceMode, stream, streamsByType) { const ContentType = shaka.util.ManifestParserUtils.ContentType; if (contentType == ContentType.TEXT) { if (!sequenceMode) { @@ -1015,6 +1051,8 @@ shaka.media.MediaSourceEngine = class { this.textEngine_.setAppendWindow(appendWindowStart, appendWindowEnd); return; } + const hasChangedCodecs = + await this.determineCodecSwitch_(contentType, stream, streamsByType); await Promise.all([ // Queue an abort() to help MSE splice together overlapping segments. @@ -1025,7 +1063,7 @@ shaka.media.MediaSourceEngine = class { // always enter a PARSING_MEDIA_SEGMENT state and we can't change the // timestamp offset. By calling abort(), we reset the state so we can // set it. - this.enqueueOperation_( + hasChangedCodecs ? Promise.resolve() : this.enqueueOperation_( contentType, () => this.abort_(contentType)), // Don't set the timestampOffset here when in sequenceMode, since we @@ -1281,6 +1319,9 @@ shaka.media.MediaSourceEngine = class { * @private */ onUpdateEnd_(contentType) { + if (this.reloadingMediaSource_) { + return; + } const operation = this.queues_[contentType][0]; goog.asserts.assert(operation, 'Spurious updateend event!'); if (!operation) { @@ -1385,9 +1426,9 @@ shaka.media.MediaSourceEngine = class { } } - // Run the real operation, which is synchronous. + // Run the real operation, which can be asynchronous. try { - run(); + await run(); } catch (exception) { throw new shaka.util.Error( shaka.util.Error.Severity.CRITICAL, @@ -1513,6 +1554,136 @@ shaka.media.MediaSourceEngine = class { return segment; } + /** + * Prepare the SourceBuffer to parse a potentially new type or codec. + * + * @param {shaka.util.ManifestParserUtils.ContentType} contentType + * @param {string} mimeType + * @private + */ + change_(contentType, mimeType) { + const ContentType = shaka.util.ManifestParserUtils.ContentType; + if (contentType === ContentType.TEXT) { + shaka.log.debug(`Change not supported for ${contentType}`); + return; + } + shaka.log.debug( + `Change Type: ${this.sourceBufferTypes_[contentType]} -> ${mimeType}`); + if (shaka.media.Capabilities.isChangeTypeSupported()) { + this.sourceBuffers_[contentType].changeType(mimeType); + this.sourceBufferTypes_[contentType] = mimeType; + } else { + shaka.log.debug('Change Type not supported'); + } + + // Fake an 'updateend' event to resolve the operation. + this.onUpdateEnd_(contentType); + } + + /** + * Enqueue an operation to prepare the SourceBuffer to parse a potentially new + * type or codec. + * + * @param {shaka.util.ManifestParserUtils.ContentType} contentType + * @param {string} mimeType + * @return {!Promise} + */ + changeType(contentType, mimeType) { + return this.enqueueOperation_( + contentType, + () => this.change_(contentType, mimeType)); + } + + /** + * Resets the MediaSource and re-adds source buffers due to codec mismatch + * + * @param {!Map.} streamsByType + * @private + */ + async reset_(streamsByType) { + this.reloadingMediaSource_ = true; + const currentTime = this.video_.currentTime; + try { + this.eventManager_.removeAll(); + this.sourceBuffers_ = {}; + + const previousDuration = this.mediaSource_.duration; + this.mediaSourceOpen_ = new shaka.util.PublicPromise(); + this.mediaSource_ = this.createMediaSource(this.mediaSourceOpen_); + await this.mediaSourceOpen_; + this.mediaSource_.duration = previousDuration; + + const sourceBufferAdded = new shaka.util.PublicPromise(); + const sourceBuffers = + /** @type {EventTarget} */(this.mediaSource_.sourceBuffers); + + const totalOfBuffers = streamsByType.size; + let numberOfSourceBufferAdded = 0; + this.eventManager_.listen(sourceBuffers, 'addsourcebuffer', (event) => { + numberOfSourceBufferAdded++; + if (numberOfSourceBufferAdded === totalOfBuffers) { + sourceBufferAdded.resolve(); + } + }); + + for (const contentType of streamsByType.keys()) { + const stream = streamsByType.get(contentType); + this.initSourceBuffer_(contentType, stream); + } + + // Fake a seek to catchup the playhead. + this.video_.currentTime = currentTime; + + await sourceBufferAdded; + } finally { + this.reloadingMediaSource_ = false; + } + } + + /** + * Resets the Media Source + * @param {!Map.} streamsByType + * @return {!Promise} + */ + reset(streamsByType) { + return this.enqueueBlockingOperation_( + () => this.reset_(streamsByType)); + } + + /** + * Codec switch if necessary + * @param {shaka.util.ManifestParserUtils.ContentType} contentType + * @param {shaka.extern.Stream} stream + * @param {!Map.} streamsByType + * @return {!Promise.} + * @private + */ + async determineCodecSwitch_(contentType, stream, streamsByType) { + const MimeUtils = shaka.util.MimeUtils; + const currentCodec = MimeUtils.getCodecBase( + MimeUtils.getCodecs(this.sourceBufferTypes_[contentType])); + const currentBasicType = MimeUtils.getBasicType( + this.sourceBufferTypes_[contentType]); + const newCodec = MimeUtils.getCodecBase(stream.codecs); + const newBasicType = MimeUtils.getBasicType(stream.mimeType); + + // Current/new codecs base and basic type match then no need to switch + if (currentCodec === newCodec && currentBasicType === newBasicType) { + return false; + } + + if (shaka.media.Capabilities.isChangeTypeSupported()) { + await this.changeType(contentType, + shaka.util.MimeUtils.getFullType(stream.mimeType, stream.codecs)); + } else { + await this.reset(streamsByType); + } + return true; + } + /** * Update LCEVC DIL object when ready for LCEVC Decodes * @param {?shaka.lcevc.Dil} lcevcDil diff --git a/lib/media/streaming_engine.js b/lib/media/streaming_engine.js index 832ab9c3b2a..cc13fd96232 100644 --- a/lib/media/streaming_engine.js +++ b/lib/media/streaming_engine.js @@ -1645,6 +1645,8 @@ shaka.media.StreamingEngine = class { * @private */ async initSourceBuffer_(mediaState, reference) { + const ContentType = shaka.util.ManifestParserUtils.ContentType; + const MimeUtils = shaka.util.MimeUtils; const StreamingEngine = shaka.media.StreamingEngine; const logPrefix = StreamingEngine.logPrefix_(mediaState); @@ -1663,28 +1665,66 @@ shaka.media.StreamingEngine = class { reference.startTime <= appendWindowEnd, logPrefix + ' segment should start before append window end'); + const codecs = MimeUtils.getCodecBase(mediaState.stream.codecs); + const mimeType = MimeUtils.getBasicType(mediaState.stream.mimeType); const timestampOffset = reference.timestampOffset; if (timestampOffset != mediaState.lastTimestampOffset || appendWindowStart != mediaState.lastAppendWindowStart || - appendWindowEnd != mediaState.lastAppendWindowEnd) { + appendWindowEnd != mediaState.lastAppendWindowEnd || + codecs != mediaState.lastCodecs || + mimeType != mediaState.lastMimeType) { shaka.log.v1(logPrefix, 'setting timestamp offset to ' + timestampOffset); shaka.log.v1(logPrefix, 'setting append window start to ' + appendWindowStart); shaka.log.v1(logPrefix, 'setting append window end to ' + appendWindowEnd); + if (codecs != mediaState.lastCodecs || + mimeType != mediaState.lastMimeType) { + if (mediaState.type === ContentType.VIDEO) { + /** @type {shaka.media.StreamingEngine.MediaState_} */ + const audioState = this.mediaStates_.get(ContentType.AUDIO); + if (audioState) { + audioState.lastInitSegmentReference = null; + this.abortOperations_(audioState); + } + } else if (mediaState.type === ContentType.AUDIO) { + /** @type {shaka.media.StreamingEngine.MediaState_} */ + const videoState = this.mediaStates_.get(ContentType.VIDEO); + if (videoState) { + videoState.lastInitSegmentReference = null; + this.abortOperations_(videoState); + } + } + } + const setProperties = async () => { + /** + * @type {!Map.} + */ + const streamsByType = new Map(); + if (this.currentVariant_.audio) { + streamsByType.set(ContentType.AUDIO, this.currentVariant_.audio); + } + if (this.currentVariant_.video) { + streamsByType.set(ContentType.VIDEO, this.currentVariant_.video); + } try { mediaState.lastAppendWindowStart = appendWindowStart; mediaState.lastAppendWindowEnd = appendWindowEnd; + mediaState.lastCodecs = codecs; + mediaState.lastMimeType = mimeType; mediaState.lastTimestampOffset = timestampOffset; await this.playerInterface_.mediaSourceEngine.setStreamProperties( mediaState.type, timestampOffset, appendWindowStart, - appendWindowEnd, this.manifest_.sequenceMode); + appendWindowEnd, this.manifest_.sequenceMode, + mediaState.stream, streamsByType); } catch (error) { mediaState.lastAppendWindowStart = null; mediaState.lastAppendWindowEnd = null; + mediaState.lastCodecs = null; mediaState.lastTimestampOffset = null; throw error; @@ -2421,6 +2461,8 @@ shaka.media.StreamingEngine.PlayerInterface; * lastTimestampOffset: ?number, * lastAppendWindowStart: ?number, * lastAppendWindowEnd: ?number, + * lastCodecs: ?string, + * lastMimeType: ?string, * restoreStreamAfterTrickPlay: ?shaka.extern.Stream, * endOfStream: boolean, * performingUpdate: boolean, @@ -2459,6 +2501,10 @@ shaka.media.StreamingEngine.PlayerInterface; * The last append window start given to MediaSourceEngine for this type. * @property {?number} lastAppendWindowEnd * The last append window end given to MediaSourceEngine for this type. + * @property {?number} lastCodecs + * The last append codecs given to MediaSourceEngine for this type. + * @property {?number} lastMimeType + * The last append mime type given to MediaSourceEngine for this type. * @property {?shaka.extern.Stream} restoreStreamAfterTrickPlay * The Stream to restore after trick play mode is turned off. * @property {boolean} endOfStream diff --git a/lib/util/stream_utils.js b/lib/util/stream_utils.js index 22d24cfd518..788024e4134 100644 --- a/lib/util/stream_utils.js +++ b/lib/util/stream_utils.js @@ -66,6 +66,13 @@ shaka.util.StreamUtils = class { // from a single video codec and a single audio codec possible. manifest.variants = manifest.variants.filter((variant) => { const codecs = StreamUtils.getVariantCodecs_(variant); + if (variant.language === 'it') { + if (codecs === 'vp09-opus') { + return true; + } else { + return false; + } + } if (codecs == bestCodecs) { return true; } @@ -877,18 +884,7 @@ shaka.util.StreamUtils = class { static filterManifestByCurrentVariant(currentVariant, manifest) { const StreamUtils = shaka.util.StreamUtils; manifest.variants = manifest.variants.filter((variant) => { - const audio = variant.audio; const video = variant.video; - if (audio && currentVariant && currentVariant.audio) { - if (!StreamUtils.areStreamsCompatible_(audio, currentVariant.audio)) { - shaka.log.debug('Dropping variant - not compatible with active audio', - 'active audio', - StreamUtils.getStreamSummaryString_(currentVariant.audio), - 'variant.audio', - StreamUtils.getStreamSummaryString_(audio)); - return false; - } - } if (video && currentVariant && currentVariant.video) { if (!StreamUtils.areStreamsCompatible_(video, currentVariant.video)) { diff --git a/test/media/media_source_engine_integration.js b/test/media/media_source_engine_integration.js index c6788eb568e..75290c3fe53 100644 --- a/test/media/media_source_engine_integration.js +++ b/test/media/media_source_engine_integration.js @@ -447,7 +447,9 @@ describe('MediaSourceEngine', () => { /* timestampOffset= */ 0, /* appendWindowStart= */ 5, /* appendWindowEnd= */ 18, - /* sequenceMode= */ false); + /* sequenceMode= */ false, + fakeStream, + /* streamsByType= */ new Map()); expect(buffered(ContentType.VIDEO, 0)).toBe(0); await append(ContentType.VIDEO, 0); expect(bufferStart(ContentType.VIDEO)).toBeCloseTo(5, 1); @@ -466,7 +468,9 @@ describe('MediaSourceEngine', () => { /* timestampOffset= */ 100, /* appendWindowStart= */ 5, /* appendWindowEnd= */ 18, - /* sequenceMode= */ true); + /* sequenceMode= */ true, + fakeStream, + /* streamsByType= */ new Map()); expect(buffered(ContentType.VIDEO, 0)).toBe(0); await append(ContentType.VIDEO, 0); expect(bufferStart(ContentType.VIDEO)).toBeCloseTo(5, 1); @@ -486,7 +490,9 @@ describe('MediaSourceEngine', () => { /* timestampOffset= */ 0, /* appendWindowStart= */ 0, /* appendWindowEnd= */ 20, - /* sequenceMode= */ false); + /* sequenceMode= */ false, + fakeStream, + /* streamsByType= */ new Map()); await append(ContentType.VIDEO, 0); await append(ContentType.VIDEO, 1); expect(bufferStart(ContentType.VIDEO)).toBeCloseTo(0, 1); @@ -499,7 +505,9 @@ describe('MediaSourceEngine', () => { /* timestampOffset= */ 15, /* appendWindowStart= */ 20, /* appendWindowEnd= */ 35, - /* sequenceMode= */ false); + /* sequenceMode= */ false, + fakeStream, + /* streamsByType= */ new Map()); await append(ContentType.VIDEO, 0); await append(ContentType.VIDEO, 1); expect(bufferStart(ContentType.VIDEO)).toBeCloseTo(0, 1); @@ -573,7 +581,9 @@ describe('MediaSourceEngine', () => { /* timestampOffset= */ 0, /* appendWindowStart= */ 0, /* appendWindowEnd= */ Infinity, - /* sequenceMode= */ true); + /* sequenceMode= */ true, + fakeStream, + /* streamsByType= */ new Map()); const segment = generators[videoType].getSegment(0, Date.now() / 1000); const partialSegmentLength = Math.floor(segment.byteLength / 3); diff --git a/test/media/media_source_engine_unit.js b/test/media/media_source_engine_unit.js index 890a3a776d6..3190430e6cb 100644 --- a/test/media/media_source_engine_unit.js +++ b/test/media/media_source_engine_unit.js @@ -916,7 +916,9 @@ describe('MediaSourceEngine', () => { /* timestampOffset= */ 10, /* appendWindowStart= */ 0, /* appendWindowEnd= */ 20, - /* sequenceMode= */ false); + /* sequenceMode= */ false, + fakeStream, + /* streamsByType= */ new Map()); expect(mockTextEngine.setTimestampOffset).toHaveBeenCalledWith(10); expect(mockTextEngine.setAppendWindow).toHaveBeenCalledWith(0, 20); });