diff --git a/content/media/test/Makefile.in b/content/media/test/Makefile.in
index d19837cd6a7e..b584c6759da5 100644
--- a/content/media/test/Makefile.in
+++ b/content/media/test/Makefile.in
@@ -343,6 +343,9 @@ ifdef MOZ_DASH
MOCHITEST_FILES += \
test_can_play_type_dash.html \
dash/dash-manifest.mpd \
+ dash/dash-manifest-sjs.mpd \
+ test_dash_detect_stream_switch.html \
+ dash_detect_stream_switch.sjs \
dash/dash-webm-video-320x180.webm \
dash/dash-webm-video-428x240.webm \
dash/dash-webm-audio-128k.webm \
diff --git a/content/media/test/dash/dash-manifest-sjs.mpd b/content/media/test/dash/dash-manifest-sjs.mpd
new file mode 100644
index 000000000000..c7ecba3c692d
--- /dev/null
+++ b/content/media/test/dash/dash-manifest-sjs.mpd
@@ -0,0 +1,35 @@
+
+
+ ./dash_detect_stream_switch.sjs?name=
+
+
+
+ dash-webm-video-320x180.webm
+
+
+
+
+
+ dash-webm-video-428x240.webm
+
+
+
+
+
+
+
+ dash-webm-audio-128k.webm
+
+
+
+
+
+
+
diff --git a/content/media/test/dash_detect_stream_switch.sjs b/content/media/test/dash_detect_stream_switch.sjs
new file mode 100644
index 000000000000..3ceecf9c7eac
--- /dev/null
+++ b/content/media/test/dash_detect_stream_switch.sjs
@@ -0,0 +1,114 @@
+/* -*- Mode: JavaScript; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
+/* vim: set ts=8 sts=2 et sw=2 tw=80: */
+/* This Source Code Form is subject to the terms of the Mozilla Public
+ * License, v. 2.0. If a copy of the MPL was not distributed with this
+ * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
+
+/* dash_detect_stream_switch.sjs
+ *
+ * Parses requests for DASH manifests and ensures stream switching takes place
+ * by verifying the subsegments downloaded and the streams they belong to.
+ * If unexpected subsegments (byte ranges) are requested, the script will
+ * will respond with a 404.
+ */
+
+var DEBUG = false;
+
+function parseQuery(request, key) {
+ var params = request.queryString.split('&');
+ if (DEBUG) {
+ dump("DASH-SJS: request params = \"" + params + "\"\n");
+ }
+ for (var j = 0; j < params.length; ++j) {
+ var p = params[j];
+ if (p == key)
+ return true;
+ if (p.indexOf(key + "=") === 0)
+ return p.substring(key.length + 1);
+ if (p.indexOf("=") < 0 && key === "")
+ return p;
+ }
+ return false;
+}
+
+function handleRequest(request, response)
+{
+ try {
+ var name = parseQuery(request, "name");
+ var range = request.hasHeader("Range") ? request.getHeader("Range")
+ : undefined;
+
+ // Should not get request for 1st subsegment from 2nd stream, nor 2nd
+ // subsegment from 1st stream.
+ if (name == "dash-webm-video-320x180.webm" && range == "bytes=25514-32767" ||
+ name == "dash-webm-video-428x240.webm" && range == "bytes=228-35852")
+ {
+ throw "Should not request " + name + " with byte-range " + range;
+ } else {
+ var rangeSplit = range.split("=");
+ if (rangeSplit.length != 2) {
+ throw "DASH-SJS: ERROR: invalid number of tokens (" + rangeSplit.length +
+ ") delimited by \'=\' in \'Range\' header.";
+ }
+ var offsets = rangeSplit[1].split("-");
+ if (offsets.length != 2) {
+ throw "DASH-SJS: ERROR: invalid number of tokens (" + offsets.length +
+ ") delimited by \'-\' in \'Range\' header.";
+ }
+ var startOffset = parseInt(offsets[0]);
+ var endOffset = parseInt(offsets[1]);
+ var file = Components.classes["@mozilla.org/file/directory_service;1"].
+ getService(Components.interfaces.nsIProperties).
+ get("CurWorkD", Components.interfaces.nsILocalFile);
+ var fis = Components.classes['@mozilla.org/network/file-input-stream;1'].
+ createInstance(Components.interfaces.nsIFileInputStream);
+ var bis = Components.classes["@mozilla.org/binaryinputstream;1"].
+ createInstance(Components.interfaces.nsIBinaryInputStream);
+
+ var paths = "tests/content/media/test/" + name;
+ var split = paths.split("/");
+ for (var i = 0; i < split.length; ++i) {
+ file.append(split[i]);
+ }
+
+ fis.init(file, -1, -1, false);
+ // Exception: start offset should be within file bounds.
+ if (startOffset > file.fileSize) {
+ throw "Starting offset [" + startOffset + "] is after end of file [" +
+ file.fileSize + "].";
+ }
+ // End offset may be too large in the MPD. Real world HTTP servers just
+ // return what data they can; do the same here - reduce the end offset.
+ if (endOffset >= file.fileSize) {
+ if (DEBUG) {
+ dump("DASH-SJS: reducing endOffset [" + endOffset + "] to fileSize [" +
+ (file.fileSize-1) + "]\n");
+ }
+ endOffset = file.fileSize-1;
+ }
+ fis.seek(Components.interfaces.nsISeekableStream.NS_SEEK_SET, startOffset);
+ bis.setInputStream(fis);
+
+ var byteLengthToRead = endOffset + 1 - startOffset;
+ var totalBytesExpected = byteLengthToRead + startOffset;
+ if (DEBUG) {
+ dump("DASH-SJS: byteLengthToRead = " + byteLengthToRead +
+ " byteLengthToRead+startOffset = " + totalBytesExpected +
+ " fileSize = " + file.fileSize + "\n");
+ }
+
+ var bytes = bis.readBytes(byteLengthToRead);
+ response.setStatusLine(request.httpVersion, 206, "Partial Content");
+ response.setHeader("Content-Length", ""+bytes.length, false);
+ response.setHeader("Content-Type", "application/dash+xml", false);
+ var contentRange = "bytes " + startOffset + "-" + endOffset + "/" +
+ file.fileSize;
+ response.setHeader("Content-Range", contentRange, false);
+ response.write(bytes, bytes.length);
+ bis.close();
+ }
+ } catch (e) {
+ dump ("DASH-SJS-ERROR: " + e + "\n");
+ response.setStatusLine(request.httpVersion, 404, "Not found");
+ }
+}
diff --git a/content/media/test/manifest.js b/content/media/test/manifest.js
index c662b5ad4610..2513049b45fd 100644
--- a/content/media/test/manifest.js
+++ b/content/media/test/manifest.js
@@ -195,6 +195,13 @@ var gInvalidTests = [
{ name:"invalid-cmap-s1c2.opus", type:"audio/ogg; codecs=opus"},
];
+// Files to test for stream switching. Note: media files referenced in DASH MPD
+// files should be accessed via dash_detect_stream_switch.sjs.
+var gStreamSwitchTests = [
+ { name:"dash-manifest-sjs.mpd", type:"application/dash+xml",
+ width:320, height:180, duration:3.966 }
+];
+
// Converts a path/filename to a file:// URI which we can load from disk.
// Optionally checks whether the file actually exists on disk at the location
// we've specified.
diff --git a/content/media/test/test_dash_detect_stream_switch.html b/content/media/test/test_dash_detect_stream_switch.html
new file mode 100644
index 000000000000..cc7152a2a2b1
--- /dev/null
+++ b/content/media/test/test_dash_detect_stream_switch.html
@@ -0,0 +1,81 @@
+
+
+
+
+ Test for Bug 792935 - DASH Stream Switching
+
+
+
+
+
+Mozilla Bug 792935
+
+
+
+
+
+
+