Bug 481057 - Fire MEDIA_ERR_DECODE errors when a decoding error occurs. r=chris.double, sr=roc

--HG--
extra : rebase_source : a07687f16726086dd425eef8907788019beebbe1
This commit is contained in:
Matthew Gregan 2009-09-22 12:08:13 +12:00
parent 9f16f9a4ac
commit 466c733803
14 changed files with 269 additions and 58 deletions

View File

@ -118,6 +118,10 @@ public:
// when the resource has a network error during loading. // when the resource has a network error during loading.
void NetworkError(); void NetworkError();
// Called by the video decoder object, on the main thread, when the
// resource has a decode error during metadata loading or decoding.
void DecodeError();
// Called by the video decoder object, on the main thread, // Called by the video decoder object, on the main thread,
// when the video playback has ended. // when the video playback has ended.
void PlaybackEnded(); void PlaybackEnded();
@ -309,11 +313,11 @@ protected:
nsresult NewURIFromString(const nsAutoString& aURISpec, nsIURI** aURI); nsresult NewURIFromString(const nsAutoString& aURISpec, nsIURI** aURI);
/** /**
* Called when all postential resources are exhausted. Changes network * Called when all potential resources are exhausted. Changes network
* state to NETWORK_NO_SOURCE, and sends error event with code * state to NETWORK_NO_SOURCE, and sends error event with code
* MEDIA_ERR_NONE_SUPPORTED. * MEDIA_ERR_SRC_NOT_SUPPORTED.
*/ */
void NoSupportedMediaError(); void NoSupportedMediaSourceError();
/** /**
* Attempts to load resources from the <source> children. This is a * Attempts to load resources from the <source> children. This is a

View File

@ -234,7 +234,7 @@ NS_IMETHODIMP nsHTMLMediaElement::MediaLoadListener::OnStartRequest(nsIRequest*
// error. // error.
if (NS_FAILED(rv) && !mNextListener && element) { if (NS_FAILED(rv) && !mNextListener && element) {
// Load failed, attempt to load the next candidate resource. If there // Load failed, attempt to load the next candidate resource. If there
// are none, this will trigger a MEDIA_ERR_NONE_SUPPORTED error. // are none, this will trigger a MEDIA_ERR_SRC_NOT_SUPPORTED error.
element->NotifyLoadError(); element->NotifyLoadError();
} }
// If InitializeDecoderForChannel did not return a listener (but may // If InitializeDecoderForChannel did not return a listener (but may
@ -407,9 +407,9 @@ void nsHTMLMediaElement::AbortExistingLoads()
mIsRunningSelectResource = PR_FALSE; mIsRunningSelectResource = PR_FALSE;
} }
void nsHTMLMediaElement::NoSupportedMediaError() void nsHTMLMediaElement::NoSupportedMediaSourceError()
{ {
mError = new nsHTMLMediaError(nsIDOMHTMLMediaError::MEDIA_ERR_NONE_SUPPORTED); mError = new nsHTMLMediaError(nsIDOMHTMLMediaError::MEDIA_ERR_SRC_NOT_SUPPORTED);
mNetworkState = nsIDOMHTMLMediaElement::NETWORK_NO_SOURCE; mNetworkState = nsIDOMHTMLMediaElement::NETWORK_NO_SOURCE;
DispatchAsyncProgressEvent(NS_LITERAL_STRING("error")); DispatchAsyncProgressEvent(NS_LITERAL_STRING("error"));
ChangeDelayLoadStatus(PR_FALSE); ChangeDelayLoadStatus(PR_FALSE);
@ -479,7 +479,7 @@ void nsHTMLMediaElement::SelectResource()
if (NS_SUCCEEDED(rv)) if (NS_SUCCEEDED(rv))
return; return;
} }
NoSupportedMediaError(); NoSupportedMediaSourceError();
} else { } else {
// Otherwise, the source elements will be used. // Otherwise, the source elements will be used.
LoadFromSourceChildren(); LoadFromSourceChildren();
@ -489,7 +489,7 @@ void nsHTMLMediaElement::SelectResource()
void nsHTMLMediaElement::NotifyLoadError() void nsHTMLMediaElement::NotifyLoadError()
{ {
if (mIsLoadingFromSrcAttribute) { if (mIsLoadingFromSrcAttribute) {
NoSupportedMediaError(); NoSupportedMediaSourceError();
} else { } else {
QueueLoadFromSourceTask(); QueueLoadFromSourceTask();
} }
@ -506,7 +506,7 @@ void nsHTMLMediaElement::LoadFromSourceChildren()
// Exhausted candidates, wait for more candidates to be appended to // Exhausted candidates, wait for more candidates to be appended to
// the media element. // the media element.
mLoadWaitStatus = WAITING_FOR_SOURCE; mLoadWaitStatus = WAITING_FOR_SOURCE;
NoSupportedMediaError(); NoSupportedMediaSourceError();
return; return;
} }
@ -1370,6 +1370,16 @@ void nsHTMLMediaElement::NetworkError()
ChangeDelayLoadStatus(PR_FALSE); ChangeDelayLoadStatus(PR_FALSE);
} }
void nsHTMLMediaElement::DecodeError()
{
mError = new nsHTMLMediaError(nsIDOMHTMLMediaError::MEDIA_ERR_DECODE);
mBegun = PR_FALSE;
DispatchAsyncProgressEvent(NS_LITERAL_STRING("error"));
mNetworkState = nsIDOMHTMLMediaElement::NETWORK_EMPTY;
DispatchAsyncSimpleEvent(NS_LITERAL_STRING("emptied"));
ChangeDelayLoadStatus(PR_FALSE);
}
void nsHTMLMediaElement::PlaybackEnded() void nsHTMLMediaElement::PlaybackEnded()
{ {
NS_ASSERTION(mDecoder->IsEnded(), "Decoder fired ended, but not in ended state"); NS_ASSERTION(mDecoder->IsEnded(), "Decoder fired ended, but not in ended state");

View File

@ -746,7 +746,7 @@ void nsOggDecodeStateMachine::HandleDecodeErrors(OggPlayErrorCode aErrorCode)
aErrorCode != E_OGGPLAY_CONTINUE) { aErrorCode != E_OGGPLAY_CONTINUE) {
mState = DECODER_STATE_SHUTDOWN; mState = DECODER_STATE_SHUTDOWN;
nsCOMPtr<nsIRunnable> event = nsCOMPtr<nsIRunnable> event =
NS_NEW_RUNNABLE_METHOD(nsOggDecoder, mDecoder, NetworkError); NS_NEW_RUNNABLE_METHOD(nsOggDecoder, mDecoder, DecodeError);
NS_DispatchToMainThread(event, NS_DISPATCH_NORMAL); NS_DispatchToMainThread(event, NS_DISPATCH_NORMAL);
} }
} }
@ -1764,7 +1764,7 @@ void nsOggDecodeStateMachine::LoadOggHeaders(nsChannelReader* aReader)
LOG(PR_LOG_DEBUG, ("Frame rate: %f", mFramerate)); LOG(PR_LOG_DEBUG, ("Frame rate: %f", mFramerate));
int aspectd, aspectn; int aspectd, aspectn;
// this can return E_OGGPLAY_UNINITIALIZED if the video has // this can return E_OGGPLAY_UNINITIALISED if the video has
// no aspect ratio data. We assume 1.0 in that case. // no aspect ratio data. We assume 1.0 in that case.
OggPlayErrorCode r = OggPlayErrorCode r =
oggplay_get_video_aspect_ratio(mPlayer, i, &aspectd, &aspectn); oggplay_get_video_aspect_ratio(mPlayer, i, &aspectd, &aspectn);
@ -1785,6 +1785,12 @@ void nsOggDecodeStateMachine::LoadOggHeaders(nsChannelReader* aReader)
} }
} }
if (mVideoTrack == -1 && mAudioTrack == -1) {
nsAutoMonitor mon(mDecoder->GetMonitor());
HandleDecodeErrors(E_OGGPLAY_UNINITIALISED);
return;
}
SetTracksActive(); SetTracksActive();
if (mVideoTrack == -1) { if (mVideoTrack == -1) {
@ -2162,6 +2168,17 @@ void nsOggDecoder::NetworkError()
Shutdown(); Shutdown();
} }
void nsOggDecoder::DecodeError()
{
if (mShuttingDown)
return;
if (mElement)
mElement->DecodeError();
Shutdown();
}
PRBool nsOggDecoder::IsSeeking() const PRBool nsOggDecoder::IsSeeking() const
{ {
return mPlayState == PLAY_STATE_SEEKING || mNextState == PLAY_STATE_SEEKING; return mPlayState == PLAY_STATE_SEEKING || mNextState == PLAY_STATE_SEEKING;

View File

@ -472,6 +472,9 @@ private:
void RegisterShutdownObserver(); void RegisterShutdownObserver();
void UnregisterShutdownObserver(); void UnregisterShutdownObserver();
// Notifies the element that decoding has failed.
void DecodeError();
/****** /******
* The following members should be accessed with the decoder lock held. * The following members should be accessed with the decoder lock held.
******/ ******/

View File

@ -87,6 +87,7 @@ _TEST_FILES = \
test_constants.html \ test_constants.html \
test_controls.html \ test_controls.html \
test_currentTime.html \ test_currentTime.html \
test_decode_error.html \
test_decoder_disable.html \ test_decoder_disable.html \
test_load.html \ test_load.html \
test_media_selection.html \ test_media_selection.html \
@ -115,6 +116,7 @@ _TEST_FILES += \
320x240.allow-origin.ogv \ 320x240.allow-origin.ogv \
320x240.allow-origin.ogv^headers^ \ 320x240.allow-origin.ogv^headers^ \
448636.ogv \ 448636.ogv \
bogus.ogv \
bug495129.ogv \ bug495129.ogv \
bug461281.ogg \ bug461281.ogg \
bug482461.ogv \ bug482461.ogv \
@ -128,6 +130,7 @@ _TEST_FILES += \
bug504613.ogv \ bug504613.ogv \
bug504644.ogv \ bug504644.ogv \
bug506094.ogv \ bug506094.ogv \
dirac.ogv \
seek.ogv \ seek.ogv \
short-video.ogv \ short-video.ogv \
small-shot.ogg \ small-shot.ogg \
@ -137,6 +140,8 @@ _TEST_FILES += \
# Wave sample files # Wave sample files
_TEST_FILES += \ _TEST_FILES += \
big.wav \ big.wav \
bogus.wav \
r11025_msadpcm_c1.wav \
r11025_s16_c1.wav \ r11025_s16_c1.wav \
r11025_s16_c1_trailing.wav \ r11025_s16_c1_trailing.wav \
r11025_u8_c1.wav \ r11025_u8_c1.wav \
@ -144,6 +149,11 @@ _TEST_FILES += \
r16000_u8_c1_list.wav \ r16000_u8_c1_list.wav \
$(NULL) $(NULL)
# Other files
_TEST_FILES += \
bogus.duh \
$(NULL)
# These tests need to be converted to be backend-independent. This list # These tests need to be converted to be backend-independent. This list
# is deprecated, do not add to it. # is deprecated, do not add to it.
ifdef MOZ_OGG ifdef MOZ_OGG

Binary file not shown.

View File

@ -11,10 +11,10 @@ var gSmallTests = [
{ name:"bogus.duh", type:"bogus/duh" } { name:"bogus.duh", type:"bogus/duh" }
]; ];
// These are files that we just want to make sure we can play through. // These are files that must fire an error during load or playback, and do not
// We can also check metadata. // cause a crash. Put files of the same type together in this list so if
// Put files of the same type together in this list so if something crashes // something crashes we have some idea of which backend is responsible. Used
// we have some idea of which backend is responsible. // by test_playback_errors, which expects one error event and no ended event.
var gPlayTests = [ var gPlayTests = [
// 8-bit samples // 8-bit samples
{ name:"r11025_u8_c1.wav", type:"audio/x-wav", duration:1.0 }, { name:"r11025_u8_c1.wav", type:"audio/x-wav", duration:1.0 },
@ -71,6 +71,20 @@ var gAudioTests = [
{ name:"bogus.duh", type:"bogus/duh", duration:123 } { name:"bogus.duh", type:"bogus/duh", duration:123 }
]; ];
// These are files suitable for testing various decoder failures that are
// expected to fire MEDIA_ERR_DECODE. Used by test_decode_error, which expects
// an error and emptied event, and no loadedmetadata or ended event.
var gDecodeErrorTests = [
// Valid files with unsupported codecs
{ name:"r11025_msadpcm_c1.wav", type:"audio/x-wav" },
{ name:"dirac.ogg", type:"video/ogg" },
// Invalid files
{ name:"bogus.wav", type:"audio/x-wav" },
{ name:"bogus.ogv", type:"video/ogg" },
{ name:"bogus.duh", type:"bogus/duh" }
];
function checkMetadata(msg, e, test) { function checkMetadata(msg, e, test) {
if (test.width) { if (test.width) {
is(e.videoWidth, test.width, msg + " video width"); is(e.videoWidth, test.width, msg + " video width");

Binary file not shown.

View File

@ -18,129 +18,226 @@ is(HTMLElement.NETWORK_EMPTY, undefined);
is(HTMLElement.NETWORK_IDLE, undefined); is(HTMLElement.NETWORK_IDLE, undefined);
is(HTMLElement.NETWORK_LOADING, undefined); is(HTMLElement.NETWORK_LOADING, undefined);
is(HTMLElement.NETWORK_LOADED, undefined); is(HTMLElement.NETWORK_LOADED, undefined);
is(HTMLElement.NETWORK_NO_SOURCE, undefined);
is(HTMLElement.HAVE_NOTHING, undefined); is(HTMLElement.HAVE_NOTHING, undefined);
is(HTMLElement.HAVE_METADATA, undefined); is(HTMLElement.HAVE_METADATA, undefined);
is(HTMLElement.HAVE_CURRENT_DATA, undefined); is(HTMLElement.HAVE_CURRENT_DATA, undefined);
is(HTMLElement.HAVE_FUTURE_DATA, undefined); is(HTMLElement.HAVE_FUTURE_DATA, undefined);
is(HTMLElement.HAVE_ENOUGH_DATA, undefined); is(HTMLElement.HAVE_ENOUGH_DATA, undefined);
is(HTMLElement.MEDIA_ERR_ABORTED, undefined);
is(HTMLElement.MEDIA_ERR_NETWORK, undefined);
is(HTMLElement.MEDIA_ERR_DECODE, undefined);
is(HTMLElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(HTMLMediaElement.NETWORK_EMPTY, 0); is(HTMLMediaElement.NETWORK_EMPTY, 0);
is(HTMLMediaElement.NETWORK_IDLE, 1); is(HTMLMediaElement.NETWORK_IDLE, 1);
is(HTMLMediaElement.NETWORK_LOADING, 2); is(HTMLMediaElement.NETWORK_LOADING, 2);
is(HTMLMediaElement.NETWORK_LOADED, 3); is(HTMLMediaElement.NETWORK_LOADED, 3);
is(HTMLMediaElement.NETWORK_NO_SOURCE, 4);
is(HTMLMediaElement.HAVE_NOTHING, 0); is(HTMLMediaElement.HAVE_NOTHING, 0);
is(HTMLMediaElement.HAVE_METADATA, 1); is(HTMLMediaElement.HAVE_METADATA, 1);
is(HTMLMediaElement.HAVE_CURRENT_DATA, 2); is(HTMLMediaElement.HAVE_CURRENT_DATA, 2);
is(HTMLMediaElement.HAVE_FUTURE_DATA, 3); is(HTMLMediaElement.HAVE_FUTURE_DATA, 3);
is(HTMLMediaElement.HAVE_ENOUGH_DATA, 4); is(HTMLMediaElement.HAVE_ENOUGH_DATA, 4);
is(HTMLMediaElement.MEDIA_ERR_ABORTED, undefined);
is(HTMLMediaElement.MEDIA_ERR_NETWORK, undefined);
is(HTMLMediaElement.MEDIA_ERR_DECODE, undefined);
is(HTMLMediaElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(HTMLVideoElement.NETWORK_EMPTY, undefined); is(HTMLVideoElement.NETWORK_EMPTY, undefined);
is(HTMLVideoElement.NETWORK_IDLE, undefined); is(HTMLVideoElement.NETWORK_IDLE, undefined);
is(HTMLVideoElement.NETWORK_LOADING, undefined); is(HTMLVideoElement.NETWORK_LOADING, undefined);
is(HTMLVideoElement.NETWORK_LOADED, undefined); is(HTMLVideoElement.NETWORK_LOADED, undefined);
is(HTMLVideoElement.NETWORK_NO_SOURCE, undefined);
is(HTMLVideoElement.HAVE_NOTHING, undefined); is(HTMLVideoElement.HAVE_NOTHING, undefined);
is(HTMLVideoElement.HAVE_METADATA, undefined); is(HTMLVideoElement.HAVE_METADATA, undefined);
is(HTMLVideoElement.HAVE_CURRENT_DATA, undefined); is(HTMLVideoElement.HAVE_CURRENT_DATA, undefined);
is(HTMLVideoElement.HAVE_FUTURE_DATA, undefined); is(HTMLVideoElement.HAVE_FUTURE_DATA, undefined);
is(HTMLVideoElement.HAVE_ENOUGH_DATA, undefined); is(HTMLVideoElement.HAVE_ENOUGH_DATA, undefined);
is(HTMLVideoElement.MEDIA_ERR_ABORTED, undefined);
is(HTMLVideoElement.MEDIA_ERR_NETWORK, undefined);
is(HTMLVideoElement.MEDIA_ERR_DECODE, undefined);
is(HTMLVideoElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(HTMLAudioElement.NETWORK_EMPTY, undefined); is(HTMLAudioElement.NETWORK_EMPTY, undefined);
is(HTMLAudioElement.NETWORK_IDLE, undefined); is(HTMLAudioElement.NETWORK_IDLE, undefined);
is(HTMLAudioElement.NETWORK_LOADING, undefined); is(HTMLAudioElement.NETWORK_LOADING, undefined);
is(HTMLAudioElement.NETWORK_LOADED, undefined); is(HTMLAudioElement.NETWORK_LOADED, undefined);
is(HTMLAudioElement.NETWORK_NO_SOURCE, undefined);
is(HTMLAudioElement.HAVE_NOTHING, undefined); is(HTMLAudioElement.HAVE_NOTHING, undefined);
is(HTMLAudioElement.HAVE_METADATA, undefined); is(HTMLAudioElement.HAVE_METADATA, undefined);
is(HTMLAudioElement.HAVE_CURRENT_DATA, undefined); is(HTMLAudioElement.HAVE_CURRENT_DATA, undefined);
is(HTMLAudioElement.HAVE_FUTURE_DATA, undefined); is(HTMLAudioElement.HAVE_FUTURE_DATA, undefined);
is(HTMLAudioElement.HAVE_ENOUGH_DATA, undefined); is(HTMLAudioElement.HAVE_ENOUGH_DATA, undefined);
is(HTMLAudioElement.MEDIA_ERR_ABORTED, undefined);
is(HTMLAudioElement.MEDIA_ERR_NETWORK, undefined);
is(HTMLAudioElement.MEDIA_ERR_DECODE, undefined);
is(HTMLAudioElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(HTMLSourceElement.NETWORK_EMPTY, undefined); is(HTMLSourceElement.NETWORK_EMPTY, undefined);
is(HTMLSourceElement.NETWORK_IDLE, undefined); is(HTMLSourceElement.NETWORK_IDLE, undefined);
is(HTMLSourceElement.NETWORK_LOADING, undefined); is(HTMLSourceElement.NETWORK_LOADING, undefined);
is(HTMLSourceElement.NETWORK_LOADED, undefined); is(HTMLSourceElement.NETWORK_LOADED, undefined);
is(HTMLSourceElement.NETWORK_NO_SOURCE, undefined);
is(HTMLSourceElement.HAVE_NOTHING, undefined); is(HTMLSourceElement.HAVE_NOTHING, undefined);
is(HTMLSourceElement.HAVE_METADATA, undefined); is(HTMLSourceElement.HAVE_METADATA, undefined);
is(HTMLSourceElement.HAVE_CURRENT_DATA, undefined); is(HTMLSourceElement.HAVE_CURRENT_DATA, undefined);
is(HTMLSourceElement.HAVE_FUTURE_DATA, undefined); is(HTMLSourceElement.HAVE_FUTURE_DATA, undefined);
is(HTMLSourceElement.HAVE_ENOUGH_DATA, undefined); is(HTMLSourceElement.HAVE_ENOUGH_DATA, undefined);
is(HTMLSourceElement.MEDIA_ERR_ABORTED, undefined);
is(HTMLSourceElement.MEDIA_ERR_NETWORK, undefined);
is(HTMLSourceElement.MEDIA_ERR_DECODE, undefined);
is(HTMLSourceElement.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(HTMLMediaError.NETWORK_EMPTY, undefined);
is(HTMLMediaError.NETWORK_IDLE, undefined);
is(HTMLMediaError.NETWORK_LOADING, undefined);
is(HTMLMediaError.NETWORK_LOADED, undefined);
is(HTMLMediaError.NETWORK_NO_SOURCE, undefined);
is(HTMLMediaError.HAVE_NOTHING, undefined);
is(HTMLMediaError.HAVE_METADATA, undefined);
is(HTMLMediaError.HAVE_CURRENT_DATA, undefined);
is(HTMLMediaError.HAVE_FUTURE_DATA, undefined);
is(HTMLMediaError.HAVE_ENOUGH_DATA, undefined);
is(HTMLMediaError.MEDIA_ERR_ABORTED, 1);
is(HTMLMediaError.MEDIA_ERR_NETWORK, 2);
is(HTMLMediaError.MEDIA_ERR_DECODE, 3);
is(HTMLMediaError.MEDIA_ERR_SRC_NOT_SUPPORTED, 4);
is(document.body.NETWORK_EMPTY, undefined); is(document.body.NETWORK_EMPTY, undefined);
is(document.body.NETWORK_IDLE, undefined); is(document.body.NETWORK_IDLE, undefined);
is(document.body.NETWORK_LOADING, undefined); is(document.body.NETWORK_LOADING, undefined);
is(document.body.NETWORK_LOADED, undefined); is(document.body.NETWORK_LOADED, undefined);
is(document.body.NETWORK_NO_SOURCE, undefined);
is(document.body.HAVE_NOTHING, undefined); is(document.body.HAVE_NOTHING, undefined);
is(document.body.HAVE_METADATA, undefined); is(document.body.HAVE_METADATA, undefined);
is(document.body.HAVE_CURRENT_DATA, undefined); is(document.body.HAVE_CURRENT_DATA, undefined);
is(document.body.HAVE_FUTURE_DATA, undefined); is(document.body.HAVE_FUTURE_DATA, undefined);
is(document.body.HAVE_ENOUGH_DATA, undefined); is(document.body.HAVE_ENOUGH_DATA, undefined);
is(document.body.MEDIA_ERR_ABORTED, undefined);
is(document.body.MEDIA_ERR_NETWORK, undefined);
is(document.body.MEDIA_ERR_DECODE, undefined);
is(document.body.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(document.getElementsByTagName("video")[0].NETWORK_EMPTY, 0); is(document.getElementsByTagName("video")[0].NETWORK_EMPTY, 0);
is(document.getElementsByTagName("video")[0].NETWORK_IDLE, 1); is(document.getElementsByTagName("video")[0].NETWORK_IDLE, 1);
is(document.getElementsByTagName("video")[0].NETWORK_LOADING, 2); is(document.getElementsByTagName("video")[0].NETWORK_LOADING, 2);
is(document.getElementsByTagName("video")[0].NETWORK_LOADED, 3); is(document.getElementsByTagName("video")[0].NETWORK_LOADED, 3);
is(document.getElementsByTagName("video")[0].NETWORK_NO_SOURCE, 4);
is(document.getElementsByTagName("video")[0].HAVE_NOTHING, 0); is(document.getElementsByTagName("video")[0].HAVE_NOTHING, 0);
is(document.getElementsByTagName("video")[0].HAVE_METADATA, 1); is(document.getElementsByTagName("video")[0].HAVE_METADATA, 1);
is(document.getElementsByTagName("video")[0].HAVE_CURRENT_DATA, 2); is(document.getElementsByTagName("video")[0].HAVE_CURRENT_DATA, 2);
is(document.getElementsByTagName("video")[0].HAVE_FUTURE_DATA, 3); is(document.getElementsByTagName("video")[0].HAVE_FUTURE_DATA, 3);
is(document.getElementsByTagName("video")[0].HAVE_ENOUGH_DATA, 4); is(document.getElementsByTagName("video")[0].HAVE_ENOUGH_DATA, 4);
is(document.getElementsByTagName("video")[0].MEDIA_ERR_ABORTED, undefined);
is(document.getElementsByTagName("video")[0].MEDIA_ERR_NETWORK, undefined);
is(document.getElementsByTagName("video")[0].MEDIA_ERR_DECODE, undefined);
is(document.getElementsByTagName("video")[0].MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(document.getElementsByTagName("audio")[0].NETWORK_EMPTY, 0); is(document.getElementsByTagName("audio")[0].NETWORK_EMPTY, 0);
is(document.getElementsByTagName("audio")[0].NETWORK_IDLE, 1); is(document.getElementsByTagName("audio")[0].NETWORK_IDLE, 1);
is(document.getElementsByTagName("audio")[0].NETWORK_LOADING, 2); is(document.getElementsByTagName("audio")[0].NETWORK_LOADING, 2);
is(document.getElementsByTagName("audio")[0].NETWORK_LOADED, 3); is(document.getElementsByTagName("audio")[0].NETWORK_LOADED, 3);
is(document.getElementsByTagName("audio")[0].NETWORK_NO_SOURCE, 4);
is(document.getElementsByTagName("audio")[0].HAVE_NOTHING, 0); is(document.getElementsByTagName("audio")[0].HAVE_NOTHING, 0);
is(document.getElementsByTagName("audio")[0].HAVE_METADATA, 1); is(document.getElementsByTagName("audio")[0].HAVE_METADATA, 1);
is(document.getElementsByTagName("audio")[0].HAVE_CURRENT_DATA, 2); is(document.getElementsByTagName("audio")[0].HAVE_CURRENT_DATA, 2);
is(document.getElementsByTagName("audio")[0].HAVE_FUTURE_DATA, 3); is(document.getElementsByTagName("audio")[0].HAVE_FUTURE_DATA, 3);
is(document.getElementsByTagName("audio")[0].HAVE_ENOUGH_DATA, 4); is(document.getElementsByTagName("audio")[0].HAVE_ENOUGH_DATA, 4);
is(document.getElementsByTagName("audio")[0].MEDIA_ERR_ABORTED, undefined);
is(document.getElementsByTagName("audio")[0].MEDIA_ERR_NETWORK, undefined);
is(document.getElementsByTagName("audio")[0].MEDIA_ERR_DECODE, undefined);
is(document.getElementsByTagName("audio")[0].MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(document.getElementsByTagName("source")[0].NETWORK_EMPTY, undefined); is(document.getElementsByTagName("source")[0].NETWORK_EMPTY, undefined);
is(document.getElementsByTagName("source")[0].NETWORK_IDLE, undefined); is(document.getElementsByTagName("source")[0].NETWORK_IDLE, undefined);
is(document.getElementsByTagName("source")[0].NETWORK_LOADING, undefined); is(document.getElementsByTagName("source")[0].NETWORK_LOADING, undefined);
is(document.getElementsByTagName("source")[0].NETWORK_LOADED, undefined); is(document.getElementsByTagName("source")[0].NETWORK_LOADED, undefined);
is(document.getElementsByTagName("source")[0].NETWORK_NO_SOURCE, undefined);
is(document.getElementsByTagName("source")[0].HAVE_NOTHING, undefined); is(document.getElementsByTagName("source")[0].HAVE_NOTHING, undefined);
is(document.getElementsByTagName("source")[0].HAVE_METADATA, undefined); is(document.getElementsByTagName("source")[0].HAVE_METADATA, undefined);
is(document.getElementsByTagName("source")[0].HAVE_CURRENT_DATA, undefined); is(document.getElementsByTagName("source")[0].HAVE_CURRENT_DATA, undefined);
is(document.getElementsByTagName("source")[0].HAVE_FUTURE_DATA, undefined); is(document.getElementsByTagName("source")[0].HAVE_FUTURE_DATA, undefined);
is(document.getElementsByTagName("source")[0].HAVE_ENOUGH_DATA, undefined); is(document.getElementsByTagName("source")[0].HAVE_ENOUGH_DATA, undefined);
is(document.getElementsByTagName("source")[0].MEDIA_ERR_ABORTED, undefined);
is(document.getElementsByTagName("source")[0].MEDIA_ERR_NETWORK, undefined);
is(document.getElementsByTagName("source")[0].MEDIA_ERR_DECODE, undefined);
is(document.getElementsByTagName("source")[0].MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(HTMLElement.prototype.NETWORK_EMPTY, undefined); is(HTMLElement.prototype.NETWORK_EMPTY, undefined);
is(HTMLElement.prototype.NETWORK_IDLE, undefined); is(HTMLElement.prototype.NETWORK_IDLE, undefined);
is(HTMLElement.prototype.NETWORK_LOADING, undefined); is(HTMLElement.prototype.NETWORK_LOADING, undefined);
is(HTMLElement.prototype.NETWORK_LOADED, undefined); is(HTMLElement.prototype.NETWORK_LOADED, undefined);
is(HTMLElement.prototype.NETWORK_NO_SOURCE, undefined);
is(HTMLElement.prototype.HAVE_NOTHING, undefined); is(HTMLElement.prototype.HAVE_NOTHING, undefined);
is(HTMLElement.prototype.HAVE_METADATA, undefined); is(HTMLElement.prototype.HAVE_METADATA, undefined);
is(HTMLElement.prototype.HAVE_CURRENT_DATA, undefined); is(HTMLElement.prototype.HAVE_CURRENT_DATA, undefined);
is(HTMLElement.prototype.HAVE_FUTURE_DATA, undefined); is(HTMLElement.prototype.HAVE_FUTURE_DATA, undefined);
is(HTMLElement.prototype.HAVE_ENOUGH_DATA, undefined); is(HTMLElement.prototype.HAVE_ENOUGH_DATA, undefined);
is(HTMLElement.prototype.MEDIA_ERR_ABORTED, undefined);
is(HTMLElement.prototype.MEDIA_ERR_NETWORK, undefined);
is(HTMLElement.prototype.MEDIA_ERR_DECODE, undefined);
is(HTMLElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
todo_is(HTMLMediaElement.prototype.NETWORK_EMPTY, 0, "HTMLMediaElement.prototype.NETWORK_EMPTY"); todo_is(HTMLMediaElement.prototype.NETWORK_EMPTY, 0, "HTMLMediaElement.prototype.NETWORK_EMPTY");
todo_is(HTMLMediaElement.prototype.NETWORK_IDLE, 1, "HTMLMediaElement.prototype.NETWORK_IDLE"); todo_is(HTMLMediaElement.prototype.NETWORK_IDLE, 1, "HTMLMediaElement.prototype.NETWORK_IDLE");
todo_is(HTMLMediaElement.prototype.NETWORK_LOADING, 2, "HTMLMediaElement.prototype.NETWORK_LOADING"); todo_is(HTMLMediaElement.prototype.NETWORK_LOADING, 2, "HTMLMediaElement.prototype.NETWORK_LOADING");
todo_is(HTMLMediaElement.prototype.NETWORK_LOADED, 3, "HTMLMediaElement.prototype.NETWORK_LOADED"); todo_is(HTMLMediaElement.prototype.NETWORK_LOADED, 3, "HTMLMediaElement.prototype.NETWORK_LOADED");
todo_is(HTMLMediaElement.prototype.NETWORK_NO_SOURCE, 4, "HTMLMediaElement.prototype.NETWORK_NO_SOURCE");
todo_is(HTMLMediaElement.prototype.HAVE_NOTHING, 0, "HTMLMediaElement.prototype.HAVE_NOTHING"); todo_is(HTMLMediaElement.prototype.HAVE_NOTHING, 0, "HTMLMediaElement.prototype.HAVE_NOTHING");
todo_is(HTMLMediaElement.prototype.HAVE_METADATA, 1, "HTMLMediaElement.prototype.HAVE_METADATA"); todo_is(HTMLMediaElement.prototype.HAVE_METADATA, 1, "HTMLMediaElement.prototype.HAVE_METADATA");
todo_is(HTMLMediaElement.prototype.HAVE_CURRENT_DATA, 2, "HTMLMediaElement.prototype.HAVE_CURRENT_DATA"); todo_is(HTMLMediaElement.prototype.HAVE_CURRENT_DATA, 2, "HTMLMediaElement.prototype.HAVE_CURRENT_DATA");
todo_is(HTMLMediaElement.prototype.HAVE_FUTURE_DATA, 3, "HTMLMediaElement.prototype.HAVE_FUTURE_DATA"); todo_is(HTMLMediaElement.prototype.HAVE_FUTURE_DATA, 3, "HTMLMediaElement.prototype.HAVE_FUTURE_DATA");
todo_is(HTMLMediaElement.prototype.HAVE_ENOUGH_DATA, 4, "HTMLMediaElement.prototype.HAVE_ENOUGH_DATA"); todo_is(HTMLMediaElement.prototype.HAVE_ENOUGH_DATA, 4, "HTMLMediaElement.prototype.HAVE_ENOUGH_DATA");
is(HTMLMediaElement.prototype.MEDIA_ERR_ABORTED, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_ABORTED");
is(HTMLMediaElement.prototype.MEDIA_ERR_NETWORK, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_NETWORK");
is(HTMLMediaElement.prototype.MEDIA_ERR_DECODE, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_DECODE");
is(HTMLMediaElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined, "HTMLMediaElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED");
is(HTMLVideoElement.prototype.NETWORK_EMPTY, 0); is(HTMLVideoElement.prototype.NETWORK_EMPTY, 0);
is(HTMLVideoElement.prototype.NETWORK_IDLE, 1); is(HTMLVideoElement.prototype.NETWORK_IDLE, 1);
is(HTMLVideoElement.prototype.NETWORK_LOADING, 2); is(HTMLVideoElement.prototype.NETWORK_LOADING, 2);
is(HTMLVideoElement.prototype.NETWORK_LOADED, 3); is(HTMLVideoElement.prototype.NETWORK_LOADED, 3);
is(HTMLVideoElement.prototype.NETWORK_NO_SOURCE, 4);
is(HTMLVideoElement.prototype.HAVE_NOTHING, 0); is(HTMLVideoElement.prototype.HAVE_NOTHING, 0);
is(HTMLVideoElement.prototype.HAVE_METADATA, 1); is(HTMLVideoElement.prototype.HAVE_METADATA, 1);
is(HTMLVideoElement.prototype.HAVE_CURRENT_DATA, 2); is(HTMLVideoElement.prototype.HAVE_CURRENT_DATA, 2);
is(HTMLVideoElement.prototype.HAVE_FUTURE_DATA, 3); is(HTMLVideoElement.prototype.HAVE_FUTURE_DATA, 3);
is(HTMLVideoElement.prototype.HAVE_ENOUGH_DATA, 4); is(HTMLVideoElement.prototype.HAVE_ENOUGH_DATA, 4);
is(HTMLVideoElement.prototype.MEDIA_ERR_ABORTED, undefined);
is(HTMLVideoElement.prototype.MEDIA_ERR_NETWORK, undefined);
is(HTMLVideoElement.prototype.MEDIA_ERR_DECODE, undefined);
is(HTMLVideoElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(HTMLAudioElement.prototype.NETWORK_EMPTY, 0); is(HTMLAudioElement.prototype.NETWORK_EMPTY, 0);
is(HTMLAudioElement.prototype.NETWORK_IDLE, 1); is(HTMLAudioElement.prototype.NETWORK_IDLE, 1);
is(HTMLAudioElement.prototype.NETWORK_LOADING, 2); is(HTMLAudioElement.prototype.NETWORK_LOADING, 2);
is(HTMLAudioElement.prototype.NETWORK_LOADED, 3); is(HTMLAudioElement.prototype.NETWORK_LOADED, 3);
is(HTMLAudioElement.prototype.NETWORK_NO_SOURCE, 4);
is(HTMLAudioElement.prototype.HAVE_NOTHING, 0); is(HTMLAudioElement.prototype.HAVE_NOTHING, 0);
is(HTMLAudioElement.prototype.HAVE_METADATA, 1); is(HTMLAudioElement.prototype.HAVE_METADATA, 1);
is(HTMLAudioElement.prototype.HAVE_CURRENT_DATA, 2); is(HTMLAudioElement.prototype.HAVE_CURRENT_DATA, 2);
is(HTMLAudioElement.prototype.HAVE_FUTURE_DATA, 3); is(HTMLAudioElement.prototype.HAVE_FUTURE_DATA, 3);
is(HTMLAudioElement.prototype.HAVE_ENOUGH_DATA, 4); is(HTMLAudioElement.prototype.HAVE_ENOUGH_DATA, 4);
is(HTMLAudioElement.prototype.MEDIA_ERR_ABORTED, undefined);
is(HTMLAudioElement.prototype.MEDIA_ERR_NETWORK, undefined);
is(HTMLAudioElement.prototype.MEDIA_ERR_DECODE, undefined);
is(HTMLAudioElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(HTMLSourceElement.prototype.NETWORK_EMPTY, undefined); is(HTMLSourceElement.prototype.NETWORK_EMPTY, undefined);
is(HTMLSourceElement.prototype.NETWORK_IDLE, undefined); is(HTMLSourceElement.prototype.NETWORK_IDLE, undefined);
is(HTMLSourceElement.prototype.NETWORK_LOADING, undefined); is(HTMLSourceElement.prototype.NETWORK_LOADING, undefined);
is(HTMLSourceElement.prototype.NETWORK_LOADED, undefined); is(HTMLSourceElement.prototype.NETWORK_LOADED, undefined);
is(HTMLSourceElement.prototype.NETWORK_NO_SOURCE, undefined);
is(HTMLSourceElement.prototype.HAVE_NOTHING, undefined); is(HTMLSourceElement.prototype.HAVE_NOTHING, undefined);
is(HTMLSourceElement.prototype.HAVE_METADATA, undefined); is(HTMLSourceElement.prototype.HAVE_METADATA, undefined);
is(HTMLSourceElement.prototype.HAVE_CURRENT_DATA, undefined); is(HTMLSourceElement.prototype.HAVE_CURRENT_DATA, undefined);
is(HTMLSourceElement.prototype.HAVE_FUTURE_DATA, undefined); is(HTMLSourceElement.prototype.HAVE_FUTURE_DATA, undefined);
is(HTMLSourceElement.prototype.HAVE_ENOUGH_DATA, undefined); is(HTMLSourceElement.prototype.HAVE_ENOUGH_DATA, undefined);
is(HTMLSourceElement.prototype.MEDIA_ERR_ABORTED, undefined);
is(HTMLSourceElement.prototype.MEDIA_ERR_NETWORK, undefined);
is(HTMLSourceElement.prototype.MEDIA_ERR_DECODE, undefined);
is(HTMLSourceElement.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, undefined);
is(HTMLMediaError.prototype.NETWORK_EMPTY, undefined);
is(HTMLMediaError.prototype.NETWORK_IDLE, undefined);
is(HTMLMediaError.prototype.NETWORK_LOADING, undefined);
is(HTMLMediaError.prototype.NETWORK_LOADED, undefined);
is(HTMLMediaError.prototype.NETWORK_NO_SOURCE, undefined);
is(HTMLMediaError.prototype.HAVE_NOTHING, undefined);
is(HTMLMediaError.prototype.HAVE_METADATA, undefined);
is(HTMLMediaError.prototype.HAVE_CURRENT_DATA, undefined);
is(HTMLMediaError.prototype.HAVE_FUTURE_DATA, undefined);
is(HTMLMediaError.prototype.HAVE_ENOUGH_DATA, undefined);
is(HTMLMediaError.prototype.MEDIA_ERR_ABORTED, 1);
is(HTMLMediaError.prototype.MEDIA_ERR_NETWORK, 2);
is(HTMLMediaError.prototype.MEDIA_ERR_DECODE, 3);
is(HTMLMediaError.prototype.MEDIA_ERR_SRC_NOT_SUPPORTED, 4);
</script> </script>
</pre> </pre>
</body> </body>

View File

@ -0,0 +1,61 @@
<!DOCTYPE HTML>
<html>
<head>
<title>Media test: unknown/invalid formats raise decode error</title>
<script type="text/javascript" src="/MochiKit/packed.js"></script>
<script type="text/javascript" src="/tests/SimpleTest/SimpleTest.js"></script>
<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css" />
<script type="text/javascript" src="use_large_cache.js"></script>
<script type="text/javascript" src="manifest.js"></script>
</head>
<body>
<pre id="test">
<script class="testbody" type="text/javascript">
var testsWaiting = 0;
gDecodeErrorTests.forEach(function (test, index) {
var v1 = document.createElement("video");
if (!v1.canPlayType(test.type)) {
return;
}
v1.addEventListener("error", function (event) {
event.stopPropagation();
var el = event.currentTarget;
is(event.type, "error", "Expected event of type 'error'");
ok(el.error, "Element 'error' attr expected to have a value");
ok(el.error instanceof HTMLMediaError, "Element 'error' attr expected to be HTMLMediaError");
is(el.error.code, HTMLMediaError.MEDIA_ERR_DECODE, "Expected a decode error");
is(el.networkState, HTMLMediaElement.NETWORK_EMPTY, "networkState should be EMPTY");
el._sawError = true;
}, false);
v1.addEventListener("emptied", function (event) {
var el = event.currentTarget;
is(el.networkState, HTMLMediaElement.NETWORK_EMPTY, "networkState should be EMPTY");
ok(el._sawError, "Expected error event");
testsWaiting -= 1;
if (testsWaiting == 0) {
SimpleTest.finish();
}
}, false);
v1.addEventListener("loadedmetadata", function () {
ok(false, "Unexpected loadedmetadata event");
}, false);
v1.autoplay = true;
v1.addEventListener("ended", function () {
ok(false, "Unexpected ended event");
}, false);
v1.src = test.name;
testsWaiting += 1;
v1.load();
});
SimpleTest.waitForExplicitFinish();
</script>
</pre>
</body>
</html>

View File

@ -164,8 +164,8 @@ function checkState() {
test_is(gMedia.autoplay, false, "Autoplay leaked"); test_is(gMedia.autoplay, false, "Autoplay leaked");
test_is(gMedia.controls, false, "Controls leaked"); test_is(gMedia.controls, false, "Controls leaked");
test_is(gMedia.muted, false, "muted leaked"); test_is(gMedia.muted, false, "muted leaked");
test_ok(gMedia.error==null || gMedia.error.code==HTMLMediaError.MEDIA_ERR_NONE_SUPPORTED, test_ok(gMedia.error==null || gMedia.error.code==HTMLMediaError.MEDIA_ERR_SRC_NOT_SUPPORTED,
"Error code should not exist or be NONE_SUPPORTED. gMedia.error=" + "Error code should not exist or be SRC_NOT_SUPPORTED. gMedia.error=" +
(gMedia.error ? gMedia.error.code : "null")); (gMedia.error ? gMedia.error.code : "null"));
test_is(gMedia.currentSrc, "", "Leaked currentSrc"); test_is(gMedia.currentSrc, "", "Leaked currentSrc");

View File

@ -490,25 +490,17 @@ nsWaveStateMachine::Run()
PRBool loaded = LoadRIFFChunk() && LoadFormatChunk() && FindDataOffset(); PRBool loaded = LoadRIFFChunk() && LoadFormatChunk() && FindDataOffset();
monitor.Enter(); monitor.Enter();
if (!loaded) {
ChangeState(STATE_ERROR);
}
if (mState == STATE_LOADING_METADATA) { if (mState == STATE_LOADING_METADATA) {
nsCOMPtr<nsIRunnable> event; mMetadataValid = PR_TRUE;
State newState; if (mNextState != STATE_SEEKING) {
nsCOMPtr<nsIRunnable> event = NS_NEW_RUNNABLE_METHOD(nsWaveDecoder, mDecoder, MetadataLoaded);
if (loaded) {
mMetadataValid = PR_TRUE;
if (mNextState != STATE_SEEKING) {
event = NS_NEW_RUNNABLE_METHOD(nsWaveDecoder, mDecoder, MetadataLoaded);
}
newState = mNextState;
} else {
event = NS_NEW_RUNNABLE_METHOD(nsWaveDecoder, mDecoder, MediaErrorDecode);
newState = STATE_ERROR;
}
if (event) {
NS_DispatchToMainThread(event, NS_DISPATCH_NORMAL); NS_DispatchToMainThread(event, NS_DISPATCH_NORMAL);
} }
ChangeState(newState); ChangeState(mNextState);
} }
} }
break; break;
@ -545,7 +537,7 @@ nsWaveStateMachine::Run()
TimeStamp now = TimeStamp::Now(); TimeStamp now = TimeStamp::Now();
TimeStamp lastWakeup = now - TimeStamp lastWakeup = now -
TimeDuration::FromMilliseconds(AUDIO_BUFFER_LENGTH); TimeDuration::FromMilliseconds(AUDIO_BUFFER_LENGTH);
do { do {
TimeDuration sleepTime = now - lastWakeup; TimeDuration sleepTime = now - lastWakeup;
@ -580,17 +572,17 @@ nsWaveStateMachine::Run()
if (mState != STATE_ENDED && if (mState != STATE_ENDED &&
availableOffset < mPlaybackPosition + len && availableOffset < mPlaybackPosition + len &&
!mStream->IsSuspendedByCache()) { !mStream->IsSuspendedByCache()) {
mBufferingStart = now; mBufferingStart = now;
mBufferingEndOffset = mPlaybackPosition + mBufferingEndOffset = mPlaybackPosition +
TimeToBytes(mBufferingWait.ToSeconds()); TimeToBytes(mBufferingWait.ToSeconds());
mBufferingEndOffset = PR_MAX(mPlaybackPosition + len, mBufferingEndOffset); mBufferingEndOffset = PR_MAX(mPlaybackPosition + len, mBufferingEndOffset);
mNextState = mState; mNextState = mState;
ChangeState(STATE_BUFFERING); ChangeState(STATE_BUFFERING);
nsCOMPtr<nsIRunnable> event = nsCOMPtr<nsIRunnable> event =
NS_NEW_RUNNABLE_METHOD(nsWaveDecoder, mDecoder, UpdateReadyStateForData); NS_NEW_RUNNABLE_METHOD(nsWaveDecoder, mDecoder, UpdateReadyStateForData);
NS_DispatchToMainThread(event, NS_DISPATCH_NORMAL); NS_DispatchToMainThread(event, NS_DISPATCH_NORMAL);
break; break;
} }
if (len > 0) { if (len > 0) {
@ -739,10 +731,16 @@ nsWaveStateMachine::Run()
break; break;
case STATE_ERROR: case STATE_ERROR:
monitor.Wait(); {
if (mState != STATE_SHUTDOWN) { nsCOMPtr<nsIRunnable> event = NS_NEW_RUNNABLE_METHOD(nsWaveDecoder, mDecoder, DecodeError);
NS_WARNING("Invalid state transition"); NS_DispatchToMainThread(event, NS_DISPATCH_NORMAL);
ChangeState(STATE_ERROR);
monitor.Wait();
if (mState != STATE_SHUTDOWN) {
NS_WARNING("Invalid state transition");
ChangeState(STATE_ERROR);
}
} }
break; break;
@ -1531,18 +1529,15 @@ nsWaveDecoder::UnregisterShutdownObserver()
} }
void void
nsWaveDecoder::MediaErrorDecode() nsWaveDecoder::DecodeError()
{ {
if (mShuttingDown) { if (mShuttingDown) {
return; return;
} }
#if 0
if (mElement) { if (mElement) {
mElement->MediaErrorDecode(); mElement->DecodeError();
} }
#else Shutdown();
NS_WARNING("MediaErrorDecode fired, but not implemented.");
#endif
} }
void void

View File

@ -244,8 +244,8 @@ private:
// Notifies the element that playback has completed. // Notifies the element that playback has completed.
void PlaybackEnded(); void PlaybackEnded();
// Notifies the element that metadata loading has failed. // Notifies the element that decoding has failed.
void MediaErrorDecode(); void DecodeError();
void RegisterShutdownObserver(); void RegisterShutdownObserver();
void UnregisterShutdownObserver(); void UnregisterShutdownObserver();

View File

@ -53,7 +53,7 @@ interface nsIDOMHTMLMediaError : nsISupports
const unsigned short MEDIA_ERR_DECODE = 3; const unsigned short MEDIA_ERR_DECODE = 3;
/* No suitable media resource could be found */ /* No suitable media resource could be found */
const unsigned short MEDIA_ERR_NONE_SUPPORTED = 4; const unsigned short MEDIA_ERR_SRC_NOT_SUPPORTED = 4;
readonly attribute unsigned short code; readonly attribute unsigned short code;
}; };