Merge autoland to central, a=merge
authorWes Kocher <wkocher@mozilla.com>
Mon, 17 Jul 2017 17:00:52 -0700
changeset 369289 efc0b1525edbd357818dc7195537364e76f709e7
parent 369249 e0b0865639cebc1b5afa0268a4b073fcdde0e69c (current diff)
parent 369186 bccbac6fcf204af8ee5a529a8b95ed8ddc94bb2f (diff)
child 369290 5e73b9798464c3f7106f0161dc9a49b234f42f9c
push id46631
push userkwierso@gmail.com
push dateTue, 18 Jul 2017 00:38:28 +0000
treeherderautoland@216a5bf264b2 [default view] [failures only]
perfherder[talos] [build metrics] [platform microbench] (compared to previous push)
reviewersmerge
milestone56.0a1
first release with
nightly linux32
nightly linux64
nightly mac
nightly win32
nightly win64
last release without
nightly linux32
nightly linux64
nightly mac
nightly win32
nightly win64
Merge autoland to central, a=merge MozReview-Commit-ID: 5dZhRSLgWzI
media/webrtc/signaling/src/media-conduit/OMXVideoCodec.cpp
media/webrtc/signaling/src/media-conduit/OMXVideoCodec.h
media/webrtc/signaling/src/media-conduit/WebrtcOMXH264VideoCodec.cpp
media/webrtc/signaling/src/media-conduit/WebrtcOMXH264VideoCodec.h
toolkit/components/places/tests/expiration/test_analyze_runs.js
toolkit/components/places/tests/expiration/test_outdated_analyze.js
toolkit/components/places/tests/unit/test_analyze.js
toolkit/themes/windows/global/icons/Search-glass.png
tools/lint/eslint/update
--- a/.hgignore
+++ b/.hgignore
@@ -115,16 +115,17 @@ GPATH
 ^testing/mozharness/.coverage
 ^testing/mozharness/nosetests.xml
 
 # Ignore tox generated dir
 .tox/
 
 # Ignore ESLint node_modules
 ^node_modules/
+^tools/lint/eslint/eslint-plugin-mozilla/node_modules/
 
 # Ignore talos virtualenv and tp5n files.
 # The tp5n set is supposed to be decompressed at
 # testing/talos/talos/tests/tp5n in order to run tests like tps
 # locally. Similarly, running talos requires a Python package virtual
 # environment. Both the virtual environment and tp5n files end up littering
 # the status command, so we ignore them.
 ^testing/talos/.Python
--- a/browser/base/content/browser-media.js
+++ b/browser/base/content/browser-media.js
@@ -163,17 +163,23 @@ var gEMEHandler = {
       Services.prefs.setBoolPref(firstPlayPref, true);
     } else {
       document.getElementById(anchorId).removeAttribute("firstplay");
     }
 
     let mainAction = {
       label: gNavigatorBundle.getString(btnLabelId),
       accessKey: gNavigatorBundle.getString(btnAccessKeyId),
-      callback() { openPreferences("panePrivacy", {origin: "browserMedia"}); },
+      callback() {
+        if (Services.prefs.getBoolPref("browser.preferences.useOldOrganization")) {
+          openPreferences("paneContent", {origin: "browserMedia"});
+        } else {
+          openPreferences("panePrivacy", {origin: "browserMedia"});
+        }
+      },
       dismiss: true
     };
     let options = {
       dismissed: true,
       eventCallback: aTopic => aTopic == "swapping",
       learnMoreURL: Services.urlFormatter.formatURLPref("app.support.baseURL") + "drm-content",
     };
     PopupNotifications.show(browser, "drmContentPlaying", message, anchorId, mainAction, null, options);
--- a/browser/components/extensions/ext-utils.js
+++ b/browser/components/extensions/ext-utils.js
@@ -517,16 +517,20 @@ class Tab extends TabBase {
     } else if (nativeTab.muteReason) {
       mutedInfo.reason = "extension";
       mutedInfo.extensionId = nativeTab.muteReason;
     }
 
     return mutedInfo;
   }
 
+  get lastAccessed() {
+    return this.nativeTab.lastAccessed;
+  }
+
   get pinned() {
     return this.nativeTab.pinned;
   }
 
   get active() {
     return this.nativeTab.selected;
   }
 
@@ -573,16 +577,17 @@ class Tab extends TabBase {
     let result = {
       sessionId: String(tabData.closedId),
       index: tabData.pos ? tabData.pos : 0,
       windowId: window && windowTracker.getId(window),
       highlighted: false,
       active: false,
       pinned: false,
       incognito: Boolean(tabData.state && tabData.state.isPrivate),
+      lastAccessed: tabData.state ? tabData.state.lastAccessed : tabData.lastAccessed,
     };
 
     if (extension.tabManager.hasTabPermission(tabData)) {
       let entries = tabData.state ? tabData.state.entries : tabData.entries;
       let entry = entries[entries.length - 1];
       result.url = entry.url;
       result.title = entry.title;
       if (tabData.image) {
--- a/browser/components/extensions/schemas/tabs.json
+++ b/browser/components/extensions/schemas/tabs.json
@@ -59,16 +59,17 @@
           "id": {"type": "integer", "minimum": -1, "optional": true, "description": "The ID of the tab. Tab IDs are unique within a browser session. Under some circumstances a Tab may not be assigned an ID, for example when querying foreign tabs using the $(ref:sessions) API, in which case a session ID may be present. Tab ID can also be set to $(ref:tabs.TAB_ID_NONE) for apps and devtools windows."},
           "index": {"type": "integer", "minimum": -1, "description": "The zero-based index of the tab within its window."},
           "windowId": {"type": "integer", "minimum": 0, "description": "The ID of the window the tab is contained within."},
           "openerTabId": {"unsupported": true, "type": "integer", "minimum": 0, "optional": true, "description": "The ID of the tab that opened this tab, if any. This property is only present if the opener tab still exists."},
           "selected": {"type": "boolean", "description": "Whether the tab is selected.", "deprecated": "Please use $(ref:tabs.Tab.highlighted).", "unsupported": true},
           "highlighted": {"type": "boolean", "description": "Whether the tab is highlighted. Works as an alias of active"},
           "active": {"type": "boolean", "description": "Whether the tab is active in its window. (Does not necessarily mean the window is focused.)"},
           "pinned": {"type": "boolean", "description": "Whether the tab is pinned."},
+          "lastAccessed": {"type": "integer", "optional": true, "description": "The last time the tab was accessed as the number of milliseconds since epoch."},
           "audible": {"type": "boolean", "optional": true, "description": "Whether the tab has produced sound over the past couple of seconds (but it might not be heard if also muted). Equivalent to whether the speaker audio indicator is showing."},
           "mutedInfo": {"$ref": "MutedInfo", "optional": true, "description": "Current tab muted state and the reason for the last state change."},
           "url": {"type": "string", "optional": true, "permissions": ["tabs"], "description": "The URL the tab is displaying. This property is only present if the extension's manifest includes the <code>\"tabs\"</code> permission."},
           "title": {"type": "string", "optional": true, "permissions": ["tabs"], "description": "The title of the tab. This property is only present if the extension's manifest includes the <code>\"tabs\"</code> permission."},
           "favIconUrl": {"type": "string", "optional": true, "permissions": ["tabs"], "description": "The URL of the tab's favicon. This property is only present if the extension's manifest includes the <code>\"tabs\"</code> permission. It may also be an empty string if the tab is loading."},
           "status": {"type": "string", "optional": true, "description": "Either <em>loading</em> or <em>complete</em>."},
           "incognito": {"type": "boolean", "description": "Whether the tab is in an incognito window."},
           "width": {"type": "integer", "optional": true, "description": "The width of the tab in pixels."},
--- a/browser/components/extensions/test/browser/browser-common.ini
+++ b/browser/components/extensions/test/browser/browser-common.ini
@@ -118,16 +118,17 @@ skip-if = debug || asan # Bug 1354681
 [browser_ext_tabs_executeScript.js]
 [browser_ext_tabs_executeScript_good.js]
 [browser_ext_tabs_executeScript_bad.js]
 [browser_ext_tabs_executeScript_multiple.js]
 [browser_ext_tabs_executeScript_no_create.js]
 [browser_ext_tabs_executeScript_runAt.js]
 [browser_ext_tabs_getCurrent.js]
 [browser_ext_tabs_insertCSS.js]
+[browser_ext_tabs_lastAccessed.js]
 [browser_ext_tabs_removeCSS.js]
 [browser_ext_tabs_move_array.js]
 [browser_ext_tabs_move_window.js]
 [browser_ext_tabs_move_window_multiple.js]
 [browser_ext_tabs_move_window_pinned.js]
 [browser_ext_tabs_onHighlighted.js]
 [browser_ext_tabs_onUpdated.js]
 [browser_ext_tabs_printPreview.js]
--- a/browser/components/extensions/test/browser/browser_ext_sessions_getRecentlyClosed_tabs.js
+++ b/browser/components/extensions/test/browser/browser_ext_sessions_getRecentlyClosed_tabs.js
@@ -38,42 +38,49 @@ add_task(async function test_sessions_ge
   });
 
   let win = await BrowserTestUtils.openNewBrowserWindow();
   await BrowserTestUtils.loadURI(win.gBrowser.selectedBrowser, "about:mozilla");
   await BrowserTestUtils.browserLoaded(win.gBrowser.selectedBrowser);
   let expectedTabs = [];
   let tab = win.gBrowser.selectedTab;
   expectedTabs.push(expectedTabInfo(tab, win));
+  let lastAccessedTimes = new Map();
+  lastAccessedTimes.set("about:mozilla", tab.lastAccessed);
 
   for (let url of ["about:robots", "about:buildconfig"]) {
     tab = await BrowserTestUtils.openNewForegroundTab(win.gBrowser, url);
     expectedTabs.push(expectedTabInfo(tab, win));
+    lastAccessedTimes.set(url, tab.lastAccessed);
   }
 
   await extension.startup();
 
   // Test with a closed tab.
   await BrowserTestUtils.removeTab(tab);
 
   extension.sendMessage("check-sessions");
   let recentlyClosed = await extension.awaitMessage("recentlyClosed");
   let tabInfo = recentlyClosed[0].tab;
   let expectedTab = expectedTabs.pop();
   checkTabInfo(expectedTab, tabInfo);
+  ok(tabInfo.lastAccessed > lastAccessedTimes.get(tabInfo.url),
+     "lastAccessed has been updated");
 
   // Test with a closed window containing tabs.
   await BrowserTestUtils.closeWindow(win);
 
   extension.sendMessage("check-sessions");
   recentlyClosed = await extension.awaitMessage("recentlyClosed");
   let tabInfos = recentlyClosed[0].window.tabs;
   is(tabInfos.length, 2, "Expected number of tabs in closed window.");
   for (let x = 0; x < tabInfos.length; x++) {
     checkTabInfo(expectedTabs[x], tabInfos[x]);
+    ok(tabInfos[x].lastAccessed > lastAccessedTimes.get(tabInfos[x].url),
+       "lastAccessed has been updated");
   }
 
   await extension.unload();
 
   // Test without tabs permission.
   extension = ExtensionTestUtils.loadExtension({
     manifest: {
       permissions: ["sessions"],
new file mode 100644
--- /dev/null
+++ b/browser/components/extensions/test/browser/browser_ext_tabs_lastAccessed.js
@@ -0,0 +1,42 @@
+"use strict";
+
+add_task(async function testLastAccessed() {
+  let past = Date.now();
+
+  await BrowserTestUtils.openNewForegroundTab(gBrowser, "https://example.com/?1");
+  await BrowserTestUtils.openNewForegroundTab(gBrowser, "https://example.com/?2");
+
+  let extension = ExtensionTestUtils.loadExtension({
+    manifest: {
+      permissions: ["tabs"],
+    },
+    async background() {
+      browser.test.onMessage.addListener(async function(msg, past) {
+        if (msg !== "past") {
+          return;
+        }
+
+        let [tab1] = await browser.tabs.query({url: "https://example.com/?1"});
+        let [tab2] = await browser.tabs.query({url: "https://example.com/?2"});
+
+        browser.test.assertTrue(tab1 && tab2, "Expected tabs were found");
+
+        let now = Date.now();
+
+        browser.test.assertTrue(past < tab1.lastAccessed &&
+                                tab1.lastAccessed < tab2.lastAccessed &&
+                                tab2.lastAccessed <= now,
+                                "lastAccessed timestamps are recent and in the right order");
+
+        await browser.tabs.remove([tab1.id, tab2.id]);
+
+        browser.test.notifyPass("tabs.lastAccessed");
+      });
+    },
+  });
+
+  await extension.startup();
+  await extension.sendMessage("past", past);
+  await extension.awaitFinish("tabs.lastAccessed");
+  await extension.unload();
+});
--- a/browser/components/newtab/NewTabSearchProvider.jsm
+++ b/browser/components/newtab/NewTabSearchProvider.jsm
@@ -63,17 +63,21 @@ SearchProvider.prototype = {
   },
 
   removeFormHistory({browser}, suggestion) {
     ContentSearch.removeFormHistoryEntry({target: browser}, suggestion);
   },
 
   manageEngines(browser) {
     const browserWin = browser.ownerGlobal;
-    browserWin.openPreferences("paneGeneral", { origin: "contentSearch" });
+    if (Services.prefs.getBoolPref("browser.preferences.useOldOrganization")) {
+      browserWin.openPreferences("paneSearch", { origin: "contentSearch" });
+    } else {
+      browserWin.openPreferences("paneGeneral", { origin: "contentSearch" });
+    }
   },
 
   async asyncGetState() {
     let state = await ContentSearch.currentStateObj(true);
     return state;
   },
 
   async asyncPerformSearch({browser}, searchData) {
--- a/browser/components/nsBrowserGlue.js
+++ b/browser/components/nsBrowserGlue.js
@@ -313,17 +313,21 @@ BrowserGlue.prototype = {
     Cu.import("resource://services-sync/main.js");
     Weave.Service.scheduler.delayedAutoConnect(delay);
   },
 
   // nsIObserver implementation
   observe: function BG_observe(subject, topic, data) {
     switch (topic) {
       case "notifications-open-settings":
-        this._openPreferences("privacy", { origin: "notifOpenSettings" });
+        if (Services.prefs.getBoolPref("browser.preferences.useOldOrganization")) {
+          this._openPreferences("content", { origin: "notifOpenSettings" });
+        } else {
+          this._openPreferences("privacy", { origin: "notifOpenSettings" });
+        }
         break;
       case "prefservice:after-app-defaults":
         this._onAppDefaults();
         break;
       case "final-ui-startup":
         this._beforeUIStartup();
         break;
       case "browser-delayed-startup-finished":
--- a/browser/extensions/formautofill/content/editProfile.xhtml
+++ b/browser/extensions/formautofill/content/editProfile.xhtml
@@ -25,17 +25,17 @@
       <input id="family-name" type="text"/>
     </label>
     <label id="organization-container">
       <span>Company</span>
       <input id="organization" type="text"/>
     </label>
     <label id="street-address-container">
       <span>Street Address</span>
-      <textarea id="street-address"/>
+      <textarea id="street-address" rows="3"/>
     </label>
     <label id="address-level2-container">
       <span>City/Town</span>
       <input id="address-level2" type="text"/>
     </label>
     <label id="address-level1-container">
       <span>State/Province</span>
       <input id="address-level1" type="text"/>
--- a/browser/extensions/formautofill/content/manageProfiles.css
+++ b/browser/extensions/formautofill/content/manageProfiles.css
@@ -20,16 +20,17 @@ fieldset > legend {
   box-sizing: border-box;
   width: 100%;
   padding: 0.4em 0.7em;
   font-size: 0.9em;
   color: #808080;
   background-color: var(--in-content-box-background-hover);
   border: 1px solid var(--in-content-box-border-color);
   border-radius: 2px 2px 0 0;
+  -moz-user-select: none;
 }
 
 option:nth-child(even) {
   background-color: -moz-oddtreerow;
 }
 
 #profiles {
   font-size: 0.85em;
--- a/browser/extensions/formautofill/content/manageProfiles.js
+++ b/browser/extensions/formautofill/content/manageProfiles.js
@@ -234,17 +234,18 @@ ManageProfileDialog.prototype = {
    *
    * @param  {DOMEvent} event
    */
   handleClick(event) {
     if (event.target == this._elements.remove) {
       this.removeAddresses(this._selectedOptions.map(option => option.value));
     } else if (event.target == this._elements.add) {
       this.openEditDialog();
-    } else if (event.target == this._elements.edit) {
+    } else if (event.target == this._elements.edit ||
+               event.target.parentNode == this._elements.addresses && event.detail > 1) {
       this.openEditDialog(this._selectedOptions[0].address);
     }
   },
 
   /**
    * Handle key press events
    *
    * @param  {DOMEvent} event
@@ -269,24 +270,26 @@ ManageProfileDialog.prototype = {
 
   /**
    * Attach event listener
    */
   attachEventListeners() {
     window.addEventListener("unload", this, {once: true});
     window.addEventListener("keypress", this);
     this._elements.addresses.addEventListener("change", this);
+    this._elements.addresses.addEventListener("click", this);
     this._elements.controlsContainer.addEventListener("click", this);
     Services.obs.addObserver(this, "formautofill-storage-changed");
   },
 
   /**
    * Remove event listener
    */
   detachEventListeners() {
     window.removeEventListener("keypress", this);
     this._elements.addresses.removeEventListener("change", this);
+    this._elements.addresses.removeEventListener("click", this);
     this._elements.controlsContainer.removeEventListener("click", this);
     Services.obs.removeObserver(this, "formautofill-storage-changed");
   },
 };
 
 new ManageProfileDialog();
--- a/browser/extensions/formautofill/locale/en-US/formautofill.properties
+++ b/browser/extensions/formautofill/locale/en-US/formautofill.properties
@@ -24,10 +24,10 @@ organization = company
 tel = phone
 email = email
 # LOCALIZATION NOTE (fieldNameSeparator): This is used as a separator between categories.
 fieldNameSeparator = ,\u0020
 # LOCALIZATION NOTE (phishingWarningMessage, phishingWarningMessage2): The warning
 # text that is displayed for informing users what categories are about to be filled.
 # "%S" will be replaced with a list generated from the pre-defined categories.
 # The text would be e.g. Also fill company, phone, email
-phishingWarningMessage = Also fill %S
-phishingWarningMessage2 = Fill %S
+phishingWarningMessage = Also autofills %S
+phishingWarningMessage2 = Autofills %S
--- a/browser/extensions/formautofill/skin/shared/editProfile.css
+++ b/browser/extensions/formautofill/skin/shared/editProfile.css
@@ -21,16 +21,17 @@ label {
 }
 
 label > span {
   box-sizing: border-box;
   flex: 0 0 8em;
   padding-inline-end: 0.5em;
   align-self: center;
   text-align: end;
+  -moz-user-select: none;
 }
 
 input,
 select {
   box-sizing: border-box;
   flex: 1 0 auto;
   width: calc(50% - 8em);
 }
--- a/browser/extensions/formautofill/test/browser/browser_autocomplete_footer.js
+++ b/browser/extensions/formautofill/test/browser/browser_autocomplete_footer.js
@@ -64,24 +64,24 @@ add_task(async function test_press_enter
 
 add_task(async function test_phishing_warning() {
   await BrowserTestUtils.withNewTab({gBrowser, url: URL}, async function(browser) {
     const {autoCompletePopup, autoCompletePopup: {richlistbox: itemsBox}} = browser;
 
     await openPopupOn(browser, "#street-address");
     const warningBox = itemsBox.querySelector(".autocomplete-richlistitem:last-child")._warningTextBox;
     ok(warningBox, "Got phishing warning box");
-    await expectWarningText(browser, "Also fill company, phone, email");
+    await expectWarningText(browser, "Also autofills company, phone, email");
     await BrowserTestUtils.synthesizeKey("VK_DOWN", {}, browser);
-    await expectWarningText(browser, "Also fill company, phone, email");
+    await expectWarningText(browser, "Also autofills company, phone, email");
     await BrowserTestUtils.synthesizeKey("VK_DOWN", {}, browser);
-    await expectWarningText(browser, "Fill address");
+    await expectWarningText(browser, "Autofills address");
     await BrowserTestUtils.synthesizeKey("VK_DOWN", {}, browser);
     await BrowserTestUtils.synthesizeKey("VK_DOWN", {}, browser);
-    await expectWarningText(browser, "Also fill company, phone, email");
+    await expectWarningText(browser, "Also autofills company, phone, email");
 
     // Ensure the popup is closed before entering the next test.
     await ContentTask.spawn(browser, {}, async function() {
       content.document.getElementById("street-address").blur();
     });
     await BrowserTestUtils.waitForCondition(() => !autoCompletePopup.popupOpen);
   });
 });
--- a/browser/extensions/formautofill/test/browser/browser_first_time_use_doorhanger.js
+++ b/browser/extensions/formautofill/test/browser/browser_first_time_use_doorhanger.js
@@ -27,19 +27,18 @@ add_task(async function test_first_time_
         form.querySelector("#tel").value = "1-345-345-3456";
 
         // Wait 500ms before submission to make sure the input value applied
         await new Promise(resolve => setTimeout(resolve, 500));
         form.querySelector("input[type=submit]").click();
       });
 
       await promiseShown;
-      let notificationElement = PopupNotifications.panel.firstChild;
       // Open the panel via main button
-      notificationElement.button.click();
+      await clickDoorhangerButton(MAIN_BUTTON_INDEX);
       let tab = await tabPromise;
       ok(tab, "Privacy panel opened");
       await BrowserTestUtils.removeTab(tab);
     }
   );
 
   addresses = await getAddresses();
   is(addresses.length, 1, "Address saved");
--- a/browser/extensions/formautofill/test/browser/browser_update_doorhanger.js
+++ b/browser/extensions/formautofill/test/browser/browser_update_doorhanger.js
@@ -6,115 +6,97 @@ add_task(async function test_update_addr
   await saveAddress(TEST_ADDRESS_1);
   let addresses = await getAddresses();
   is(addresses.length, 1, "1 address in storage");
 
   await BrowserTestUtils.withNewTab({gBrowser, url: FORM_URL},
     async function(browser) {
       let promiseShown = BrowserTestUtils.waitForEvent(PopupNotifications.panel,
                                                        "popupshown");
-      await ContentTask.spawn(browser, null, async function() {
-        content.document.querySelector("form #organization").focus();
-      });
-
-      await BrowserTestUtils.synthesizeKey("VK_DOWN", {}, browser);
-      await expectPopupOpen(browser);
+      await openPopupOn(browser, "form #organization");
       await BrowserTestUtils.synthesizeKey("VK_DOWN", {}, browser);
       await BrowserTestUtils.synthesizeKey("VK_RETURN", {}, browser);
 
       await ContentTask.spawn(browser, null, async function() {
         let form = content.document.getElementById("form");
         let org = form.querySelector("#organization");
         await new Promise(resolve => setTimeout(resolve, 1000));
         org.value = "Mozilla";
 
         // Wait 1000ms before submission to make sure the input value applied
         await new Promise(resolve => setTimeout(resolve, 1000));
         form.querySelector("input[type=submit]").click();
       });
 
       await promiseShown;
-      let notificationElement = PopupNotifications.panel.firstChild;
-      notificationElement.button.click();
+      await clickDoorhangerButton(MAIN_BUTTON_INDEX);
     }
   );
 
   addresses = await getAddresses();
   is(addresses.length, 1, "Still 1 address in storage");
   is(addresses[0].organization, "Mozilla", "Verify the organization field");
 });
 
 add_task(async function test_create_new_address() {
   let addresses = await getAddresses();
   is(addresses.length, 1, "1 address in storage");
 
   await BrowserTestUtils.withNewTab({gBrowser, url: FORM_URL},
     async function(browser) {
       let promiseShown = BrowserTestUtils.waitForEvent(PopupNotifications.panel,
                                                        "popupshown");
-      await ContentTask.spawn(browser, null, async function() {
-        content.document.querySelector("form #tel").focus();
-      });
-
-      await BrowserTestUtils.synthesizeKey("VK_DOWN", {}, browser);
-      await expectPopupOpen(browser);
+      await openPopupOn(browser, "form #tel");
       await BrowserTestUtils.synthesizeKey("VK_DOWN", {}, browser);
       await BrowserTestUtils.synthesizeKey("VK_RETURN", {}, browser);
 
       await ContentTask.spawn(browser, null, async function() {
         let form = content.document.getElementById("form");
         let tel = form.querySelector("#tel");
         await new Promise(resolve => setTimeout(resolve, 1000));
         tel.value = "+1-234-567-890";
 
         // Wait 1000ms before submission to make sure the input value applied
         await new Promise(resolve => setTimeout(resolve, 1000));
         form.querySelector("input[type=submit]").click();
       });
 
       await promiseShown;
-      let notificationElement = PopupNotifications.panel.firstChild;
-      notificationElement.secondaryButton.click();
+      await clickDoorhangerButton(SECONDARY_BUTTON_INDEX);
     }
   );
 
   addresses = await getAddresses();
   is(addresses.length, 2, "2 addresses in storage");
   is(addresses[1].tel, "+1-234-567-890", "Verify the tel field");
 });
 
 add_task(async function test_create_new_address_merge() {
   let addresses = await getAddresses();
-  is(addresses.length, 2, "2 address in storage");
+  is(addresses.length, 2, "2 addresses in storage");
 
   await BrowserTestUtils.withNewTab({gBrowser, url: FORM_URL},
     async function(browser) {
       let promiseShown = BrowserTestUtils.waitForEvent(PopupNotifications.panel,
                                                        "popupshown");
-      await ContentTask.spawn(browser, null, async function() {
-        content.document.querySelector("form #tel").focus();
-      });
-
-      await BrowserTestUtils.synthesizeKey("VK_DOWN", {}, browser);
-      await expectPopupOpen(browser);
+      await openPopupOn(browser, "form #tel");
       await BrowserTestUtils.synthesizeKey("VK_DOWN", {}, browser);
       await BrowserTestUtils.synthesizeKey("VK_RETURN", {}, browser);
 
       // Choose the latest address and revert to the original phone number
       await ContentTask.spawn(browser, null, async function() {
         let form = content.document.getElementById("form");
         let tel = form.querySelector("#tel");
         tel.value = "+1 617 253 5702";
 
         // Wait 1000ms before submission to make sure the input value applied
         await new Promise(resolve => setTimeout(resolve, 1000));
         form.querySelector("input[type=submit]").click();
       });
 
       await promiseShown;
-      let notificationElement = PopupNotifications.panel.firstChild;
-      notificationElement.secondaryButton.click();
+      await clickDoorhangerButton(SECONDARY_BUTTON_INDEX);
     }
   );
 
   addresses = await getAddresses();
   is(addresses.length, 2, "Still 2 addresses in storage");
 });
--- a/browser/extensions/formautofill/test/browser/head.js
+++ b/browser/extensions/formautofill/test/browser/head.js
@@ -1,11 +1,11 @@
 /* exported MANAGE_PROFILES_DIALOG_URL, EDIT_PROFILE_DIALOG_URL, BASE_URL,
             TEST_ADDRESS_1, TEST_ADDRESS_2, TEST_ADDRESS_3,
-            sleep, expectPopupOpen, openPopupOn,
+            sleep, expectPopupOpen, openPopupOn, clickDoorhangerButton,
             getAddresses, saveAddress, removeAddresses */
 
 "use strict";
 
 const MANAGE_PROFILES_DIALOG_URL = "chrome://formautofill/content/manageProfiles.xhtml";
 const EDIT_PROFILE_DIALOG_URL = "chrome://formautofill/content/editProfile.xhtml";
 const BASE_URL = "http://mochi.test:8888/browser/browser/extensions/formautofill/test/browser/";
 
@@ -28,16 +28,19 @@ const TEST_ADDRESS_2 = {
   country: "US",
 };
 
 const TEST_ADDRESS_3 = {
   "street-address": "Other Address",
   "postal-code": "12345",
 };
 
+const MAIN_BUTTON_INDEX = 0;
+const SECONDARY_BUTTON_INDEX = 1;
+
 async function sleep(ms = 500) {
   await new Promise(resolve => setTimeout(resolve, ms));
 }
 
 async function expectPopupOpen(browser) {
   const {autoCompletePopup, autoCompletePopup: {richlistbox: itemsBox}} = browser;
   const listItemElems = itemsBox.querySelectorAll(".autocomplete-richlistitem");
 
@@ -76,14 +79,40 @@ function saveAddress(address) {
   return TestUtils.topicObserved("formautofill-storage-changed");
 }
 
 function removeAddresses(guids) {
   Services.cpmm.sendAsyncMessage("FormAutofill:RemoveAddresses", {guids});
   return TestUtils.topicObserved("formautofill-storage-changed");
 }
 
+/**
+ * Clicks the popup notification button and wait for popup hidden.
+ *
+ * @param {number} buttonIndex Number indicating which button to click.
+ *                             See the constants in this file.
+ */
+async function clickDoorhangerButton(buttonIndex) {
+  let popuphidden = BrowserTestUtils.waitForEvent(PopupNotifications.panel, "popuphidden");
+  let notifications = PopupNotifications.panel.childNodes;
+  ok(notifications.length > 0, "at least one notification displayed");
+  ok(true, notifications.length + " notification(s)");
+  let notification = notifications[0];
+
+  if (buttonIndex == MAIN_BUTTON_INDEX) {
+    ok(true, "Triggering main action");
+    EventUtils.synthesizeMouseAtCenter(notification.button, {});
+  } else if (buttonIndex == SECONDARY_BUTTON_INDEX) {
+    ok(true, "Triggering secondary action");
+    EventUtils.synthesizeMouseAtCenter(notification.secondaryButton, {});
+  } else if (notification.childNodes[buttonIndex - 1]) {
+    ok(true, "Triggering secondary action with index " + buttonIndex);
+    EventUtils.synthesizeMouseAtCenter(notification.childNodes[buttonIndex - 1], {});
+  }
+  await popuphidden;
+}
+
 registerCleanupFunction(async function() {
   let addresses = await getAddresses();
   if (addresses.length) {
     await removeAddresses(addresses.map(address => address.guid));
   }
 });
--- a/browser/extensions/onboarding/content/onboarding-tour-agent.js
+++ b/browser/extensions/onboarding/content/onboarding-tour-agent.js
@@ -10,18 +10,29 @@ document.getElementById("onboarding-over
                         .addEventListener("click", evt => {
   switch (evt.target.id) {
     case "onboarding-tour-addons-button":
       Mozilla.UITour.showHighlight("addons");
       break;
     case "onboarding-tour-customize-button":
       Mozilla.UITour.showHighlight("customize");
       break;
-   case "onboarding-tour-default-browser-button":
-      Mozilla.UITour.setConfiguration("defaultBrowser");
+    case "onboarding-tour-default-browser-button":
+      Mozilla.UITour.getConfiguration("appinfo", (config) => {
+        let isDefaultBrowser = config.defaultBrowser;
+        let btn = document.getElementById("onboarding-tour-default-browser-button");
+        let msg = document.getElementById("onboarding-tour-is-default-browser-msg");
+        if (isDefaultBrowser) {
+          btn.classList.add("onboarding-hidden");
+          msg.classList.remove("onboarding-hidden");
+        } else {
+          btn.disabled = true;
+          Mozilla.UITour.setConfiguration("defaultBrowser");
+        }
+      });
       break;
     case "onboarding-tour-private-browsing-button":
       Mozilla.UITour.showHighlight("privateWindow");
       break;
     case "onboarding-tour-search-button":
       Mozilla.UITour.openSearchPanel(() => {});
       break;
     case "onboarding-tour-sync-button":
@@ -29,14 +40,15 @@ document.getElementById("onboarding-over
       if (emailInput.checkValidity()) {
         Mozilla.UITour.showFirefoxAccounts(null, emailInput.value);
       }
       break;
     case "onboarding-overlay":
     case "onboarding-overlay-close-btn":
       // Dismiss any highlights if a user tries to close the dialog.
       Mozilla.UITour.hideHighlight();
+      break;
   }
   // Dismiss any highlights if a user tries to change to other tours.
   if (evt.target.classList.contains("onboarding-tour-item")) {
     Mozilla.UITour.hideHighlight();
   }
 });
--- a/browser/extensions/onboarding/content/onboarding.css
+++ b/browser/extensions/onboarding/content/onboarding.css
@@ -28,17 +28,18 @@
   height: 29px;
   position: absolute;
   cursor: pointer;
   top: 30px;
   offset-inline-start: 30px;
   background: url("img/overlay-icon.svg") no-repeat;
 }
 
-#onboarding-overlay-dialog {
+#onboarding-overlay-dialog,
+.onboarding-hidden {
   display: none;
 }
 
 #onboarding-overlay-close-btn,
 #onboarding-notification-close-btn {
   position: absolute;
   top: 15px;
   offset-inline-end: 15px;
@@ -144,16 +145,26 @@
   padding-inline-start: 29px;
 }
 
 #onboarding-tour-list > li.onboarding-active,
 #onboarding-tour-list > li:hover {
   color: #0A84FF;
 }
 
+/* Default browser tour */
+#onboarding-tour-is-default-browser-msg {
+  font-size: 16px;
+  line-height: 21px;
+  float: inline-end;
+  margin-inline-end: 26px;
+  margin-top: -32px;
+  text-align: center;
+}
+
 /* Sync tour */
 #onboarding-tour-sync-page form {
   text-align: center;
 }
 
 #onboarding-tour-sync-page form > h3 {
   text-align: center;
   margin: 0;
@@ -256,23 +267,23 @@
   border: none;
   border-radius: 0;
   color: #fff;
   float: inline-end;
   margin-inline-end: 26px;
   margin-top: -32px;
 }
 
-.onboarding-tour-action-button:hover:not([disabled="true"]) ,
+.onboarding-tour-action-button:hover:not([disabled]) ,
 #onboarding-notification-action-btn:hover {
   background: #0060df;
   cursor: pointer;
 }
 
-.onboarding-tour-action-button:active:not([disabled="true"]),
+.onboarding-tour-action-button:active:not([disabled]),
 #onboarding-notification-action-btn:active  {
   background: #003EAA;
 }
 
 .onboarding-tour-action-button:disabled {
   opacity: 0.5;
 }
 
--- a/browser/extensions/onboarding/content/onboarding.js
+++ b/browser/extensions/onboarding/content/onboarding.js
@@ -155,31 +155,34 @@ var onboardingTourset = {
     tourNameId: "onboarding.tour-default-browser",
     getNotificationStrings(bundle) {
       return {
         title: bundle.formatStringFromName("onboarding.notification.onboarding-tour-default-browser.title", [BRAND_SHORT_NAME], 1),
         message: bundle.formatStringFromName("onboarding.notification.onboarding-tour-default-browser.message", [BRAND_SHORT_NAME], 1),
         button: bundle.GetStringFromName("onboarding.button.learnMore"),
       };
     },
-    getPage(win) {
+    getPage(win, bundle) {
       let div = win.document.createElement("div");
       let defaultBrowserButtonId = win.matchMedia("(-moz-os-version: windows-win7)").matches ?
         "onboarding.tour-default-browser.win7.button" : "onboarding.tour-default-browser.button";
+      let isDefaultMessage = bundle.GetStringFromName("onboarding.tour-default-browser.is-default.message");
+      let isDefault2ndMessage = bundle.formatStringFromName("onboarding.tour-default-browser.is-default.2nd-message", [BRAND_SHORT_NAME], 1);
       // eslint-disable-next-line no-unsanitized/property
       div.innerHTML = `
         <section class="onboarding-tour-description">
           <h1 data-l10n-id="onboarding.tour-default-browser.title2"></h1>
           <p data-l10n-id="onboarding.tour-default-browser.description2"></p>
         </section>
         <section class="onboarding-tour-content">
           <img src="resource://onboarding/img/figure_default.svg" />
         </section>
         <aside class="onboarding-tour-button-container">
           <button id="onboarding-tour-default-browser-button" class="onboarding-tour-action-button" data-l10n-id="${defaultBrowserButtonId}"></button>
+          <div id="onboarding-tour-is-default-browser-msg" class="onboarding-hidden">${isDefaultMessage}<br/>${isDefault2ndMessage}</div>
         </aside>
       `;
       return div;
     },
   },
   "sync": {
     id: "onboarding-tour-sync",
     tourNameId: "onboarding.tour-sync2",
@@ -228,18 +231,16 @@ var onboardingTourset = {
  */
 class Onboarding {
   constructor(contentWindow) {
     this.init(contentWindow);
   }
 
   async init(contentWindow) {
     this._window = contentWindow;
-    this._tourItems = [];
-    this._tourPages = [];
     this._tours = [];
 
     let tourIds = this._getTourIDList(Services.prefs.getStringPref("browser.onboarding.tour-type", "update"));
     tourIds.forEach(tourId => {
       if (onboardingTourset[tourId]) {
         this._tours.push(onboardingTourset[tourId]);
       }
     });
@@ -275,16 +276,18 @@ class Onboarding {
     }
   }
 
   _initUI() {
     if (this.uiInitialized) {
       return;
     }
     this.uiInitialized = true;
+    this._tourItems = [];
+    this._tourPages = [];
 
     this._overlayIcon = this._renderOverlayIcon();
     this._overlayIcon.addEventListener("click", this);
     this._window.document.body.appendChild(this._overlayIcon);
 
     this._overlay = this._renderOverlay();
     this._overlay.addEventListener("click", this);
     this._window.document.body.appendChild(this._overlay);
@@ -406,16 +409,17 @@ class Onboarding {
 
     this._clearPrefObserver();
     this._overlayIcon.remove();
     this._overlay.remove();
     if (this._notificationBar) {
       this._notificationBar.remove();
     }
 
+    this._tourItems = this._tourPages =
     this._overlayIcon = this._overlay = this._notificationBar = null;
   }
 
   toggleOverlay() {
     if (this._tourItems.length == 0) {
       // Lazy loading until first toggle.
       this._loadTours(this._tours);
     }
@@ -459,17 +463,17 @@ class Onboarding {
     });
     if (params.length > 0) {
       this.sendMessageToChrome("set-prefs", params);
     }
   }
 
   markTourCompletionState(tourId) {
     // We are doing lazy load so there might be no items.
-    if (this._tourItems.length > 0 && this.isTourCompleted(tourId)) {
+    if (this._tourItems && this._tourItems.length > 0 && this.isTourCompleted(tourId)) {
       let targetItem = this._tourItems.find(item => item.id == tourId);
       targetItem.classList.add("onboarding-complete");
     }
   }
 
   _muteNotificationOnFirstSession() {
     if (Preferences.isSet("browser.onboarding.notification.tour-ids-queue")) {
       // There is a queue. We had prompted before, this must not be the 1st session.
--- a/browser/extensions/onboarding/locales/en-US/onboarding.properties
+++ b/browser/extensions/onboarding/locales/en-US/onboarding.properties
@@ -51,16 +51,20 @@ onboarding.tour-default-browser=Default 
 # LOCALIZATION NOTE(onboarding.tour-default-browser.title2): This string will be used in the default browser tour title. %S is brandShortName
 onboarding.tour-default-browser.title2=Make %S your go-to browser.
 # LOCALIZATION NOTE(onboarding.tour-default-browser.description2): This string will be used in the default browser tour description. %1$S is brandShortName
 onboarding.tour-default-browser.description2=Love %1$S? Set it as your default browser. Open a link from another application, and %1$S will be there for you.
 # LOCALIZATION NOTE(onboarding.tour-default-browser.button): Label for a button to open the OS default browser settings where it's not possible to set the default browser directly. (OSX, Linux, Windows 8 and higher)
 onboarding.tour-default-browser.button=Open Default Browser Settings
 # LOCALIZATION NOTE(onboarding.tour-default-browser.win7.button): Label for a button to directly set the default browser (Windows 7). %S is brandShortName
 onboarding.tour-default-browser.win7.button=Make %S Your Default Browser
+# LOCALIZATION NOTE(onboarding.tour-default-browser.is-default.message): Label displayed when Firefox is already set as default browser. followed on a new line by "tour-default-browser.is-default.2nd-message".
+onboarding.tour-default-browser.is-default.message=You’ve got this!
+# LOCALIZATION NOTE(onboarding.tour-default-browser.is-default.2nd-message): Label displayed when Firefox is already set as default browser. %S is brandShortName
+onboarding.tour-default-browser.is-default.2nd-message=%S is already your default browser.
 # LOCALIZATION NOTE(onboarding.notification.onboarding-tour-default-browser.title): This string will be used in the notification message for the default browser tour. %S is brandShortName.
 onboarding.notification.onboarding-tour-default-browser.title=Make %S your go-to browser.
 # LOCALIZATION NOTE(onboarding.notification.onboarding-tour-default-browser.message): %1$S is brandShortName
 onboarding.notification.onboarding-tour-default-browser.message=It doesn’t take much to get the most from %1$S. Just set %1$S as your default browser and put control, customization, and protection on autopilot.
 
 onboarding.tour-sync2=Sync
 onboarding.tour-sync.title2=Pick up where you left off.
 onboarding.tour-sync.description2=Sync makes it easy to access bookmarks, passwords, and even open tabs on all your devices. Sync also gives you control of the types of information you want, and don’t want, to share.
--- a/browser/extensions/onboarding/test/browser/head.js
+++ b/browser/extensions/onboarding/test/browser/head.js
@@ -51,17 +51,17 @@ function promiseOnboardingOverlayLoaded(
         resolve(false);
       });
     })
   };
   return BrowserTestUtils.waitForCondition(
     condition,
     "Should load onboarding overlay",
     100,
-    30
+    50 // Bug 1381335 increased retries, so debug builds can trigger idle in time
   );
 }
 
 function promiseOnboardingOverlayOpened(browser) {
   let condition = () => {
     return ContentTask.spawn(browser, {}, function() {
       return new Promise(resolve => {
         let overlay = content.document.querySelector("#onboarding-overlay");
--- a/browser/locales/search/list.json
+++ b/browser/locales/search/list.json
@@ -717,17 +717,17 @@
         "visibleDefaultEngines": [
           "google", "yahoo", "bing", "amazondotcom", "ddg", "twitter", "wikipedia-uz"
         ]
       }
     },
     "vi": {
       "default": {
         "visibleDefaultEngines": [
-          "google", "ddg", "wikipedia-vi", "zing-mp3"
+          "google", "coccoc", "ddg", "wikipedia-vi", "zing-mp3"
         ]
       }
     },
     "wo": {
       "default": {
         "visibleDefaultEngines": [
           "google", "yahoo", "bing", "amazondotcom", "ddg", "wikipedia-wo"
         ]
new file mode 100644
--- /dev/null
+++ b/browser/locales/searchplugins/coccoc.xml
@@ -0,0 +1,22 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<!-- 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/. -->
+
+<SearchPlugin xmlns="http://www.mozilla.org/2006/browser/search/">
+<ShortName>Cốc Cốc</ShortName>
+<Description>Use Cốc Cốc to search on coccoc.com</Description>
+<InputEncoding>UTF-8</InputEncoding>
+<Image width="16" height="16"></Image>
+<Url type="application/x-suggestions+json" method="GET"
+     template="https://coccoc.com/composer/autocomplete">
+  <Param name="of"  value="b" />
+  <Param name="q" value="{searchTerms}"/>
+  <Param name="s" value="ff"/>
+</Url>
+<Url type="text/html" method="GET" template="https://coccoc.com/search" resultdomain="coccoc.com">
+  <Param name="query" value="{searchTerms}"/>
+  <Param name="s" value="ff"/>
+  <Param name="utm_source" value="firefox"/>
+</Url>
+</SearchPlugin>
--- a/browser/themes/linux/syncedtabs/sidebar.css
+++ b/browser/themes/linux/syncedtabs/sidebar.css
@@ -41,17 +41,17 @@ html {
 .textbox-search-clear {
   background-image: url(moz-icon://stock/gtk-clear?size=menu);
   background-repeat: no-repeat;
   width: 16px;
   height: 16px;
 }
 
 .textbox-search-icon {
-  background-image: url(moz-icon://stock/gtk-find?size=menu);
+  background-image: url(chrome://global/skin/icons/search-textbox.svg);
   background-repeat: no-repeat;
   width: 16px;
   height: 16px;
   display: block;
 }
 
 .textbox-search-icon[searchbutton]:not([disabled]) ,
 .textbox-search-clear:not([disabled]) {
--- a/browser/themes/shared/incontentprefs/preferences.inc.css
+++ b/browser/themes/shared/incontentprefs/preferences.inc.css
@@ -12,17 +12,18 @@
   font-size: 1.25rem;
 }
 
 * {
   -moz-user-select: text;
 }
 
 button,
-treecol {
+treecol,
+html|option {
   /* override the * rule */
   -moz-user-select: none;
 }
 
 #engineList treechildren::-moz-tree-image(engineShown, checked),
 #blocklistsTree treechildren::-moz-tree-image(selectionCol, checked) {
   list-style-image: url("chrome://global/skin/in-content/check.svg");
   -moz-context-properties: fill, stroke;
index 17d27c99eede5f41ab7114e3e97762c0d6daf5b1..a5fcc1dec4abcb966b3066bb4b50db4a45bedb57
GIT binary patch
literal 984
zc$@*)11J26P)<h;3K|Lk000e1NJLTq001Tc002k`1ONa4Fv?F400002VoOIv0063u
zBQgL0010qNS#tmY3ljhU3ljkVnw%H_000McNliru;Rg;1Apx~KX$JrR18GS_K~z}7
zt(e=6+ei#VPf>E_ve=*6{|AD6N?w9&#$xS5x9%6WXGg-ohAbVEtRlNT`2G9aKd-#-
z1xHU$lK5+2ug)AWzv2HPd)_khP6D9CC;klV-GlEWaRTsDk#Xl^16#hT9oX|1AN=7?
zS05Vx@_As-yZ$Knpr1eqB;AC2DzY0vzk(S7s^I4O^}G=z!3{93uq0QJwyL>-B}pQL
z7C^T|*YgV8p~7-Hhw8d)*R$u<_6O9OEkT+POW*~%=Y`~7k{;ul-U#XzJu%kqGR+s=
zbz`Weona;U@rv4f7Sp+Qz6l`e`|(SB4I7Um*c{ggyYyUNz2$nVCTM}hqQ#mb(_;LT
zdX+Osx`ixy)TJ@K)4>^=>kkE3A6lg9nr3c*r+(WCkgHgj>b6bwM1YYcNC*;5I&SrB
zPE$3<fE+4lo8kiz=E!=m$xM*JK9oe)?KHw};M(K{ltj*6YZ76GeZx)$Destc$8l^d
zTzRHuf%%RE*cf#|q#ZS`*^do1v(qxdPCFeovM&1bRLvVERThy>gGbC!cbq6X!PH}h
zQY{aJy09~!QgnuOwS7We^qCB%J;8079hPkdBt*(#NfAV*k5s)#Q{W&7g@cGdR;iqi
zqtkX!#-h=KP$&pGI)}qiKi7rh4(C}AwYj4-ew0c#S8M^3R6vqcNI{7+X!P28Y6uYy
zge-5z=0kfdhvVGNHNx_m$I%=p8k?j`nvx<E(#864X<ET(fzc9+2xyUEg1JswghD7s
zC6~Z5)*Dx+MO>1qVO8KIR&N#@pQl+p0>Y}|R{UkPDz-mRh$+61Aj=ZE9*>6;ceSFc
zra2jPgh)ET?4peQ#zysV75FylmUcS)wr!Oj_$+pNE{!RcB>mYHeKcBaRFy5miLnm)
zGY)>nX|b40vOJ$=dAQu$0@z%Q!*ZWJ{-AQPo#>)%s<xy#wbVFyOUca6t_LwL+iC8A
zX$#&ud`{AIRq$|WY_8e4o0@CbA5QkgK6jPu1N_z+^L=wmt&{bxSI=v9n(X|r&*D3%
z*<$oORk3}|7Qefm|Nl+D4{J8aQzTtjW0&cZYOY7nSM=r{yW7HCz-+s{F1m&dPW$s?
zXRYSnJ-1)`j<%%dYxWYoR!hUduxhkP&ySz{)ZV<GIsOCIEO#x?2IFV|0000<MNUMn
GLSTZ6_0ZY?
--- a/browser/themes/shared/tabs.inc.css
+++ b/browser/themes/shared/tabs.inc.css
@@ -258,35 +258,35 @@
   padding-inline-start: var(--tab-curve-half-width);
 }
 
 /* Tab Overflow */
 .tabbrowser-arrowscrollbox > .arrowscrollbox-overflow-start-indicator:not([collapsed]),
 .tabbrowser-arrowscrollbox > .arrowscrollbox-overflow-end-indicator:not([collapsed]) {
   background-image: url(chrome://browser/skin/tabbrowser/tab-overflow-indicator.png);
   background-size: 100% 100%;
-  width: 14px;
+  width: 19px;
   margin-bottom: calc(var(--navbar-tab-toolbar-highlight-overlap) + var(--tab-toolbar-navbar-overlap));
   pointer-events: none;
   position: relative;
   z-index: 3; /* the selected tab's z-index + 1 */
 }
 
 .tabbrowser-arrowscrollbox > .arrowscrollbox-overflow-start-indicator:-moz-locale-dir(rtl),
 .tabbrowser-arrowscrollbox > .arrowscrollbox-overflow-end-indicator:-moz-locale-dir(ltr) {
   transform: scaleX(-1);
 }
 
 .tabbrowser-arrowscrollbox > .arrowscrollbox-overflow-start-indicator:not([collapsed]) {
   margin-inline-start: -2px;
-  margin-inline-end: -12px;
+  margin-inline-end: -17px;
 }
 
 .tabbrowser-arrowscrollbox > .arrowscrollbox-overflow-end-indicator:not([collapsed]) {
-  margin-inline-start: -12px;
+  margin-inline-start: -17px;
   margin-inline-end: -2px;
 }
 
 .tabbrowser-arrowscrollbox > .arrowscrollbox-overflow-start-indicator[collapsed],
 .tabbrowser-arrowscrollbox > .arrowscrollbox-overflow-end-indicator[collapsed] {
   opacity: 0;
 }
 
--- a/browser/themes/windows/syncedtabs/sidebar.css
+++ b/browser/themes/windows/syncedtabs/sidebar.css
@@ -29,17 +29,17 @@ html {
   padding: 2px 2px 3px;
   padding-inline-start: 4px;
   color: -moz-FieldText;
 }
 
 .textbox-search-icon {
   width: 16px;
   height: 16px;
-  background-image: url(chrome://global/skin/icons/Search-glass.png);
+  background-image: url(chrome://global/skin/icons/search-textbox.svg);
   background-repeat: no-repeat;
   display: block;
 }
 
 .textbox-search-icon:-moz-locale-dir(rtl) {
   transform: scaleX(-1);
 }
 
--- a/dom/media/MediaPrefs.h
+++ b/dom/media/MediaPrefs.h
@@ -119,16 +119,18 @@ private:
 #ifdef MOZ_GONK_MEDIACODEC
   DECL_MEDIA_PREF("media.gonk.enabled",                       PDMGonkDecoderEnabled, bool, true);
 #endif
 #ifdef MOZ_WIDGET_ANDROID
   DECL_MEDIA_PREF("media.android-media-codec.enabled",        PDMAndroidMediaCodecEnabled, bool, false);
   DECL_MEDIA_PREF("media.android-media-codec.preferred",      PDMAndroidMediaCodecPreferred, bool, false);
   DECL_MEDIA_PREF("media.navigator.hardware.vp8_encode.acceleration_remote_enabled", RemoteMediaCodecVP8EncoderEnabled, bool, false);
 #endif
+  // WebRTC
+  DECL_MEDIA_PREF("media.navigator.mediadatadecoder_enabled", MediaDataDecoderEnabled, bool, false);
 #ifdef MOZ_FFMPEG
   DECL_MEDIA_PREF("media.ffmpeg.enabled",                     PDMFFmpegEnabled, bool, true);
   DECL_MEDIA_PREF("media.libavcodec.allow-obsolete",          LibavcodecAllowObsolete, bool, false);
 #endif
 #if defined(MOZ_FFMPEG) || defined(MOZ_FFVPX)
   DECL_MEDIA_PREF("media.ffmpeg.low-latency.enabled",         PDMFFmpegLowLatencyEnabled, bool, false);
 #endif
 #ifdef MOZ_FFVPX
--- a/dom/media/test/mochitest.ini
+++ b/dom/media/test/mochitest.ini
@@ -1005,17 +1005,16 @@ skip-if = toolkit == 'android' # bug 129
 [test_resolution_change.html]
 skip-if = toolkit == 'android' # android(bug 1232305)
 tags=capturestream
 [test_resume.html]
 skip-if = true # bug 1021673
 [test_seek_negative.html]
 skip-if = toolkit == 'android' # bug 1295443, bug 1306787, android(bug 1232305)
 [test_seek_nosrc.html]
-skip-if = toolkit == 'android' # android(bug 1232305)
 [test_seek_out_of_range.html]
 skip-if = toolkit == 'android' # bug 1299382, android(bug 1232305)
 [test_seek_promise_bug1344357.html]
 skip-if = toolkit == 'android' # bug 1299382, android(bug 1232305)
 [test_seek-1.html]
 skip-if = toolkit == 'android' # bug 1322806, android(bug 1232305)
 [test_seek-2.html]
 skip-if = toolkit == 'android' # bug 1309778, android(bug 1232305)
new file mode 100644
--- /dev/null
+++ b/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe-ref.html
@@ -0,0 +1,4 @@
+<!DOCTYPE HTML>
+<img style="position:absolute; left:0; top:0"
+src=""
+>
new file mode 100644
--- /dev/null
+++ b/dom/media/test/reftest/bipbop_300_215kbps.mp4.lastframe.html
@@ -0,0 +1,19 @@
+<!DOCTYPE HTML>
+<html class="reftest-wait">
+<head>
+<script type="text/javascript">
+function doTest() {
+  var video = document.getElementById("v1");
+  video.src = "../bipbop_300_215kbps.mp4";
+  video.play();
+  video.addEventListener("ended", function() {
+    document.documentElement.removeAttribute('class');
+  }, {once: true});
+}
+window.addEventListener("MozReftestInvalidate", doTest);
+</script>
+</head>
+<body>
+<video id="v1" style="position:absolute; left:0; top:0"></video>
+</body>
+</html>
--- a/dom/media/test/reftest/generateREF.html
+++ b/dom/media/test/reftest/generateREF.html
@@ -1,55 +1,55 @@
 <!DOCTYPE HTML>
 <html>
 <head>
 <script type="application/javascript">
 </script>
 </head>
 <body>
+<p id="out"></p>
 <video id="v1" style="position:absolute; left:0; top:0"></video>
 <canvas id="canvas"></canvas>
 <script type="application/javascript">
 /* READ ME first.
 The script is trying to make a reftest sample for reftest.
 HOW TO USE:
 1. Choose the first or last frame you want to generate. And set
-window.onload function to dumpFirstFrameToConsole/dumpLastFrameToConsole.
-2. Set the video.src in dumpFirstFrameToConsole/dumpLastFrameToConsole.
+window.onload function to dumpFirstFrame/dumpLastFrame.
+2. Set the video.src in dumpFirstFrame/dumpLastFrame.
 3. Run the script on browser.
-4. Open console (ctrl+shift+k) to get the first/last frame.
-5. Copy the url to your xxx-ref.html(short.mp4.firstframe-ref.html).
+4. Copy the base64 image url to your xxx-ref.html(short.mp4.firstframe-ref.html).
 You might hit security error if the video.src cross origin.
 Enable "media.seekToNextFrame.enabled" for the seekToNextFrame function
 or using nightly, the seekToNextFrame() ensure the ended event fired.
 */
 
-//window.onload = function() { setTimeout(dumpFirstFrameToConsole, 0); };
-window.onload = function() { setTimeout(dumpLastFrameToConsole, 0); };
+//window.onload = function() { setTimeout(dumpFirstFrame, 0); };
+window.onload = function() { setTimeout(dumpLastFrame, 0); };
 
-function drawVideoToConsole(v) {
+function drawVideoToInnerHTML(v) {
   var canvas = document.getElementById("canvas");
   canvas.width = v.videoWidth;
   canvas.height = v.videoHeight;
   var ctx = canvas.getContext("2d");
   ctx.drawImage(v, 0, 0, v.videoWidth, v.videoHeight);
   var dataURL = canvas.toDataURL();
-  console.log(dataURL);
+  document.getElementById("out").innerHTML=dataURL;
 }
 
-function dumpFirstFrameToConsole() {
+function dumpFirstFrame() {
   var video = document.getElementById("v1");
   video.src = "short.mp4";
   video.preload = "metadata";
   video.addEventListener("loadeddata", function() {
-    drawVideoToConsole(video);
+    drawVideoToInnerHTML(video);
   });
 }
 
-function dumpLastFrameToConsole() {
+function dumpLastFrame() {
   var video = document.getElementById("v1");
   video.src = "short.mp4";
   video.preload = "metadata";
   video.seenEnded = false;
   // Seek to the end
   video.addEventListener("loadeddata", function() {
     video.currentTime = video.duration;
     video.onseeked = () => {
@@ -67,14 +67,14 @@ function dumpLastFrameToConsole() {
       () => {
         // Reach the end, do nothing.
       }
     );
   }
 
   video.addEventListener("ended", function() {
     video.seenEnded = true;
-    drawVideoToConsole(video);
+    drawVideoToInnerHTML(video);
   });
 }
 </script>
 </body>
 </html>
--- a/dom/media/test/reftest/reftest.list
+++ b/dom/media/test/reftest/reftest.list
@@ -1,2 +1,3 @@
 skip-if(Android) fuzzy-if(OSX,22,49977) skip-if(winWidget) fuzzy-if(webrender,70,600) HTTP(..) == short.mp4.firstframe.html short.mp4.firstframe-ref.html
 skip-if(Android) fuzzy-if(OSX,23,51392) fuzzy-if(winWidget,59,76797) fuzzy-if(webrender,60,1800) HTTP(..) == short.mp4.lastframe.html short.mp4.lastframe-ref.html
+skip-if(Android) skip-if(winWidget) fuzzy-if(webrender,55,4281) fuzzy-if(OSX,3,111852) HTTP(..) == bipbop_300_215kbps.mp4.lastframe.html bipbop_300_215kbps.mp4.lastframe-ref.html
\ No newline at end of file
--- a/layout/style/test/stylo-failures.md
+++ b/layout/style/test/stylo-failures.md
@@ -47,24 +47,23 @@ to mochitest command.
 * Unsupported values
   * SVG-in-OpenType values not supported servo/servo#15211 bug 1355412
     * test_value_storage.html `context-` [7]
     * test_bug798843_pref.html [3]
 * Incorrect parsing
   * different parsing bug 1364260
     * test_supports_rules.html [6]
     * test_condition_text.html [1]
-  * test_value_storage.html `calc(25% - 10%)`: calc percent in -webkit-gradient bug 1380918 [5]
 * Incorrect serialization
   * place-{content,items,self} shorthands bug 1363971
     * test_align_shorthand_serialization.html [6]
   * system font serialization with subprop specified bug 1364286
     * test_system_font_serialization.html [3]
   * radial gradients are not serialized using modern unprefixed style bug 1380259
-    * test_computed_style.html `gradient` [2]
+    * test_computed_style.html `gradient` [1]
 * Unit should be preserved after parsing servo/servo#15346
   * test_units_time.html [1]
 * getComputedStyle style doesn't contain custom properties bug 1336891
   * test_variables.html `custom property name` [2]
 * test_css_supports.html: issues around @supports syntax servo/servo#15482 [2]
 * test_author_specified_style.html: support serializing color as author specified bug 1348165 [27]
 * browser_newtab_share_rule_processors.js: agent style sheet sharing [1]
 * :visited support (bug 1328509)
--- a/media/libstagefright/binding/MP4Metadata.cpp
+++ b/media/libstagefright/binding/MP4Metadata.cpp
@@ -763,17 +763,17 @@ MP4MetadataRust::Init()
   if (MOZ_LOG_TEST(sLog, LogLevel::Debug)) {
     mp4parse_log(true);
   }
 
   mp4parse_status rv = mp4parse_read(mRustParser.get());
   MOZ_LOG(sLog, LogLevel::Debug, ("rust parser returned %d\n", rv));
   Telemetry::Accumulate(Telemetry::MEDIA_RUST_MP4PARSE_SUCCESS,
                         rv == mp4parse_status_OK);
-  if (rv != mp4parse_status_OK) {
+  if (rv != mp4parse_status_OK && rv != mp4parse_status_TABLE_TOO_LARGE) {
     MOZ_LOG(sLog, LogLevel::Info, ("Rust mp4 parser fails to parse this stream."));
     MOZ_ASSERT(rv > 0);
     Telemetry::Accumulate(Telemetry::MEDIA_RUST_MP4PARSE_ERROR_CODE, rv);
     return false;
   }
 
   UpdateCrypto();
 
--- a/media/libstagefright/binding/include/mp4parse.h
+++ b/media/libstagefright/binding/include/mp4parse.h
@@ -18,16 +18,17 @@ extern "C" {
 
 typedef enum mp4parse_status {
 	mp4parse_status_OK = 0,
 	mp4parse_status_BAD_ARG = 1,
 	mp4parse_status_INVALID = 2,
 	mp4parse_status_UNSUPPORTED = 3,
 	mp4parse_status_EOF = 4,
 	mp4parse_status_IO = 5,
+	mp4parse_status_TABLE_TOO_LARGE = 6,
 } mp4parse_status;
 
 typedef enum mp4parse_track_type {
 	mp4parse_track_type_VIDEO = 0,
 	mp4parse_track_type_AUDIO = 1,
 } mp4parse_track_type;
 
 typedef enum mp4parse_codec {
--- a/media/libstagefright/binding/mp4parse-cargo.patch
+++ b/media/libstagefright/binding/mp4parse-cargo.patch
@@ -1,55 +1,55 @@
 diff --git a/media/libstagefright/binding/mp4parse/Cargo.toml b/media/libstagefright/binding/mp4parse/Cargo.toml
 index ff9422c..814c4c6 100644
 --- a/media/libstagefright/binding/mp4parse/Cargo.toml
 +++ b/media/libstagefright/binding/mp4parse/Cargo.toml
-@@ -18,19 +18,13 @@ exclude = [
+@@ -20,19 +20,11 @@ exclude = [
  ]
  
+-[badges]
+-travis-ci = { repository = "https://github.com/mozilla/mp4parse-rust" }
+ 
  [dependencies]
 -byteorder = "1.0.0"
 -afl = { version = "0.1.1", optional = true }
 -afl-plugin = { version = "0.1.1", optional = true }
 -abort_on_panic = { version = "1.0.0", optional = true }
 -bitreader = { version = "0.3.0" }
 -num-traits = "0.1.37"
 +byteorder = "1.0.0"
 +bitreader = { version = "0.3.0" }
 +num-traits = "0.1.37"
-
+ 
  [dev-dependencies]
  test-assembler = "0.1.2"
  
 -[features]
 -fuzz = ["afl", "afl-plugin", "abort_on_panic"]
 -
  # Somewhat heavy-handed, but we want at least -Z force-overflow-checks=on.
  [profile.release]
  debug-assertions = true
 diff --git a/media/libstagefright/binding/mp4parse_capi/Cargo.toml b/media/libstagefright/binding/mp4parse_capi/Cargo.toml
 index a30e045..a965f06 100644
 --- a/media/libstagefright/binding/mp4parse_capi/Cargo.toml
 +++ b/media/libstagefright/binding/mp4parse_capi/Cargo.toml
-@@ -18,22 +18,13 @@ exclude = [
+@@ -18,19 +18,10 @@ exclude = [
    "*.mp4",
  ]
- 
+
 -build = "build.rs"
 -
 -[badges]
 -travis-ci = { repository = "https://github.com/mozilla/mp4parse-rust" }
 +build = false
- 
+
  [dependencies]
  byteorder = "1.0.0"
  mp4parse = {version = "0.8.0", path = "../mp4parse"}
  num-traits = "0.1.37"
- 
+
 -[build-dependencies]
 -moz-cheddar = "0.4.0"
 -
 -[features]
 -fuzz = ["mp4parse/fuzz"]
 -
- # Somewhat heavy-handed, but we want at least -Z force-overflow-checks=on.
- [profile.release]
- debug-assertions = true
--- a/media/libstagefright/binding/mp4parse/Cargo.toml
+++ b/media/libstagefright/binding/mp4parse/Cargo.toml
@@ -3,33 +3,28 @@ name = "mp4parse"
 version = "0.8.0"
 authors = [
   "Ralph Giles <giles@mozilla.com>",
   "Matthew Gregan <kinetik@flim.org>",
   "Alfredo Yang <ayang@mozilla.com>",
 ]
 
 description = "Parser for ISO base media file format (mp4)"
-documentation = "https://mp4parse-docs.surge.sh/mp4parse/"
+documentation = "https://docs.rs/mp4parse/"
 license = "MPL-2.0"
 categories = ["multimedia::video"]
 
 repository = "https://github.com/mozilla/mp4parse-rust"
 
 # Avoid complaints about trying to package test files.
 exclude = [
   "*.mp4",
 ]
 
-[badges]
-travis-ci = { repository = "https://github.com/mozilla/mp4parse-rust" }
 
 [dependencies]
 byteorder = "1.0.0"
 bitreader = { version = "0.3.0" }
 num-traits = "0.1.37"
 
 [dev-dependencies]
 test-assembler = "0.1.2"
 
-# Somewhat heavy-handed, but we want at least -Z force-overflow-checks=on.
-[profile.release]
-debug-assertions = true
--- a/media/libstagefright/binding/mp4parse/src/boxes.rs
+++ b/media/libstagefright/binding/mp4parse/src/boxes.rs
@@ -132,14 +132,9 @@ box_database!(
     QTWaveAtom                        0x77617665, // "wave" - quicktime atom
     ProtectionSystemSpecificHeaderBox 0x70737368, // "pssh"
     SchemeInformationBox              0x73636869, // "schi"
     TrackEncryptionBox                0x74656e63, // "tenc"
     ProtectionSchemeInformationBox    0x73696e66, // "sinf"
     OriginalFormatBox                 0x66726d61, // "frma"
     MP3AudioSampleEntry               0x2e6d7033, // ".mp3" - from F4V.
     CompositionOffsetBox              0x63747473, // "ctts"
-    JPEGAtom                          0x6a706567, // "jpeg" - QT JPEG
-    AC3SampleEntry                    0x61632d33, // "ac-3"
-    EC3SampleEntry                    0x65632d33, // "ec-3"
-    AC3SpecificBox                    0x64616333, // "dac3"
-    EC3SpecificBox                    0x64656333, // "dec3"
 );
--- a/media/libstagefright/binding/mp4parse/src/lib.rs
+++ b/media/libstagefright/binding/mp4parse/src/lib.rs
@@ -62,16 +62,18 @@ pub enum Error {
     /// Parse error caused by limited parser support rather than invalid data.
     Unsupported(&'static str),
     /// Reflect `std::io::ErrorKind::UnexpectedEof` for short data.
     UnexpectedEOF,
     /// Propagate underlying errors from `std::io`.
     Io(std::io::Error),
     /// read_mp4 terminated without detecting a moov box.
     NoMoov,
+    /// Parse error caused by table size is over limitation.
+    TableTooLarge,
 }
 
 impl From<bitreader::BitReaderError> for Error {
     fn from(_: bitreader::BitReaderError) -> Error {
         Error::InvalidData("invalid data")
     }
 }
 
@@ -262,18 +264,16 @@ pub struct ES_Descriptor {
 }
 
 #[allow(non_camel_case_types)]
 #[derive(Debug, Clone)]
 pub enum AudioCodecSpecific {
     ES_Descriptor(ES_Descriptor),
     FLACSpecificBox(FLACSpecificBox),
     OpusSpecificBox(OpusSpecificBox),
-    AC3SpecificBox,
-    EC3SpecificBox,
     MP3,
 }
 
 #[derive(Debug, Clone)]
 pub struct AudioSampleEntry {
     data_reference_index: u16,
     pub channelcount: u16,
     pub samplesize: u16,
@@ -282,17 +282,16 @@ pub struct AudioSampleEntry {
     pub protection_info: Vec<ProtectionSchemeInfoBox>,
 }
 
 #[derive(Debug, Clone)]
 pub enum VideoCodecSpecific {
     AVCConfig(Vec<u8>),
     VPxConfig(VPxConfigBox),
     ESDSConfig(Vec<u8>),
-    JPEG,
 }
 
 #[derive(Debug, Clone)]
 pub struct VideoSampleEntry {
     data_reference_index: u16,
     pub width: u16,
     pub height: u16,
     pub codec_specific: VideoCodecSpecific,
@@ -411,19 +410,16 @@ pub enum CodecType {
     Opus,
     H264,   // 14496-10
     MP4V,   // 14496-2
     VP10,
     VP9,
     VP8,
     EncryptedVideo,
     EncryptedAudio,
-    JPEG,   // QT JPEG atom
-    AC3,    // Digital Audio Compression (AC-3, Enhanced AC-3) Standard, ETSI TS 102 366.
-    EC3,    // Digital Audio Compression (AC-3, Enhanced AC-3) Standard, ETSI TS 102 366.
 }
 
 impl Default for CodecType {
     fn default() -> Self { CodecType::Unknown }
 }
 
 /// The media's global (mvhd) timescale in units per second.
 #[derive(Debug, Copy, Clone, PartialEq)]
@@ -1157,17 +1153,17 @@ fn read_stss<T: Read>(src: &mut BMFFBox<
 
 /// Parse a stsc box.
 fn read_stsc<T: Read>(src: &mut BMFFBox<T>) -> Result<SampleToChunkBox> {
     let (_, _) = read_fullbox_extra(src)?;
     let sample_count = be_u32_with_limit(src)?;
     let mut samples = Vec::new();
     for _ in 0..sample_count {
         let first_chunk = be_u32(src)?;
-        let samples_per_chunk = be_u32(src)?;
+        let samples_per_chunk = be_u32_with_limit(src)?;
         let sample_description_index = be_u32(src)?;
         samples.push(SampleToChunk {
             first_chunk: first_chunk,
             samples_per_chunk: samples_per_chunk,
             sample_description_index: sample_description_index,
         });
     }
 
@@ -1190,17 +1186,17 @@ fn read_ctts<T: Read>(src: &mut BMFFBox<
 
     let mut offsets = Vec::new();
     for _ in 0..counts {
         let (sample_count, time_offset) = match version {
             // According to spec, Version0 shoule be used when version == 0;
             // however, some buggy contents have negative value when version == 0.
             // So we always use Version1 here.
             0...1 => {
-                let count = be_u32(src)?;
+                let count = be_u32_with_limit(src)?;
                 let offset = TimeOffsetVersion::Version1(be_i32(src)?);
                 (count, offset)
             },
             _ => {
                 return Err(Error::InvalidData("unsupported version in 'ctts' box"));
             }
         };
         offsets.push(TimeOffset {
@@ -1238,17 +1234,17 @@ fn read_stsz<T: Read>(src: &mut BMFFBox<
 }
 
 /// Parse a stts box.
 fn read_stts<T: Read>(src: &mut BMFFBox<T>) -> Result<TimeToSampleBox> {
     let (_, _) = read_fullbox_extra(src)?;
     let sample_count = be_u32_with_limit(src)?;
     let mut samples = Vec::new();
     for _ in 0..sample_count {
-        let sample_count = be_u32(src)?;
+        let sample_count = be_u32_with_limit(src)?;
         let sample_delta = be_u32(src)?;
         samples.push(Sample {
             sample_count: sample_count,
             sample_delta: sample_delta,
         });
     }
 
     // Padding could be added in some contents.
@@ -1669,17 +1665,16 @@ fn read_hdlr<T: Read>(src: &mut BMFFBox<
 fn read_video_sample_entry<T: Read>(src: &mut BMFFBox<T>) -> Result<(CodecType, SampleEntry)> {
     let name = src.get_header().name;
     let codec_type = match name {
         BoxType::AVCSampleEntry | BoxType::AVC3SampleEntry => CodecType::H264,
         BoxType::MP4VideoSampleEntry => CodecType::MP4V,
         BoxType::VP8SampleEntry => CodecType::VP8,
         BoxType::VP9SampleEntry => CodecType::VP9,
         BoxType::ProtectedVisualSampleEntry => CodecType::EncryptedVideo,
-        BoxType::JPEGAtom => CodecType::JPEG,
         _ => {
             log!("Unsupported video codec, box {:?} found", name);
             CodecType::Unknown
         }
     };
 
     // Skip uninteresting fields.
     skip(src, 6)?;
@@ -1691,21 +1686,17 @@ fn read_video_sample_entry<T: Read>(src:
 
     let width = be_u16(src)?;
     let height = be_u16(src)?;
 
     // Skip uninteresting fields.
     skip(src, 50)?;
 
     // Skip clap/pasp/etc. for now.
-    let mut codec_specific = if name == BoxType::JPEGAtom {
-        Some(VideoCodecSpecific::JPEG)
-    } else {
-        None
-    };
+    let mut codec_specific = None;
     let mut protection_info = Vec::new();
     let mut iter = src.box_iter();
     while let Some(mut b) = iter.next_box()? {
         match b.head.name {
             BoxType::AVCConfigurationBox => {
                 if (name != BoxType::AVCSampleEntry &&
                     name != BoxType::AVC3SampleEntry &&
                     name != BoxType::ProtectedVisualSampleEntry) ||
@@ -1732,21 +1723,16 @@ fn read_video_sample_entry<T: Read>(src:
                 if name != BoxType::MP4VideoSampleEntry || codec_specific.is_some() {
                     return Err(Error::InvalidData("malformed video sample entry"));
                 }
                 let (_, _) = read_fullbox_extra(&mut b.content)?;
                 let esds_size = b.head.size - b.head.offset - 4;
                 let esds = read_buf(&mut b.content, esds_size as usize)?;
                 codec_specific = Some(VideoCodecSpecific::ESDSConfig(esds));
             }
-            BoxType::JPEGAtom => {
-                if name != BoxType::JPEGAtom || codec_specific.is_some() {
-                    return Err(Error::InvalidData("malformed video sample entry"));
-                }
-            }
             BoxType::ProtectionSchemeInformationBox => {
                 if name != BoxType::ProtectedVisualSampleEntry {
                     return Err(Error::InvalidData("malformed video sample entry"));
                 }
                 let sinf = read_sinf(&mut b)?;
                 log!("{:?} (sinf)", sinf);
                 protection_info.push(sinf);
             }
@@ -1870,36 +1856,16 @@ fn read_audio_sample_entry<T: Read>(src:
                 if name != BoxType::ProtectedAudioSampleEntry {
                     return Err(Error::InvalidData("malformed audio sample entry"));
                 }
                 let sinf = read_sinf(&mut b)?;
                 log!("{:?} (sinf)", sinf);
                 codec_type = CodecType::EncryptedAudio;
                 protection_info.push(sinf);
             }
-            BoxType::AC3SpecificBox => {
-                if name != BoxType::AC3SampleEntry {
-                    return Err(Error::InvalidData("malformed AC3 sample entry"));
-                }
-                // TODO: AC3SpecificBox needs to be parsed for detail information.
-                skip_box_remain(&mut b)?;
-                log!("(ac3)");
-                codec_type = CodecType::AC3;
-                codec_specific = Some(AudioCodecSpecific::AC3SpecificBox);
-            }
-            BoxType::EC3SpecificBox => {
-                if name != BoxType::EC3SpecificBox {
-                    return Err(Error::InvalidData("malformed EC3 sample entry"));
-                }
-                // TODO: EC3SpecificBox needs to be parsed for detail information.
-                skip_box_remain(&mut b)?;
-                log!("(ec3)");
-                codec_type = CodecType::EC3;
-                codec_specific = Some(AudioCodecSpecific::EC3SpecificBox);
-            }
             _ => {
                 log!("Unsupported audio codec, box {:?} found", b.head.name);
                 skip_box_content(&mut b)?;
             }
         }
         check_parser_state!(b.content);
     }
 
@@ -2079,17 +2045,17 @@ fn be_u24<T: ReadBytesExt>(src: &mut T) 
 fn be_u32<T: ReadBytesExt>(src: &mut T) -> Result<u32> {
     src.read_u32::<byteorder::BigEndian>().map_err(From::from)
 }
 
 /// Using in reading table size and return error if it exceeds limitation.
 fn be_u32_with_limit<T: ReadBytesExt>(src: &mut T) -> Result<u32> {
     be_u32(src).and_then(|v| {
         if v > TABLE_SIZE_LIMIT {
-            return Err(Error::Unsupported("Over limited value"));
+            return Err(Error::TableTooLarge);
         }
         Ok(v)
     })
 }
 
 fn be_u64<T: ReadBytesExt>(src: &mut T) -> Result<u64> {
     src.read_u64::<byteorder::BigEndian>().map_err(From::from)
 }
--- a/media/libstagefright/binding/mp4parse/src/tests.rs
+++ b/media/libstagefright/binding/mp4parse/src/tests.rs
@@ -896,41 +896,16 @@ fn read_esds() {
     assert_eq!(es.audio_object_type, Some(2));
     assert_eq!(es.audio_sample_rate, Some(24000));
     assert_eq!(es.audio_channel_count, Some(6));
     assert_eq!(es.codec_esds, aac_esds);
     assert_eq!(es.decoder_specific_data, aac_dc_descriptor);
 }
 
 #[test]
-fn read_ac3_sample_entry() {
-    let ac3 =
-        vec![
-            0x00, 0x00, 0x00, 0x0b, 0x64, 0x61, 0x63, 0x33, 0x10, 0x11, 0x60
-        ];
-
-    let mut stream = make_box(BoxSize::Auto, b"ac-3", |s| {
-        s.append_repeated(0, 6)
-         .B16(1)    // data_reference_count
-         .B16(0)
-         .append_repeated(0, 6)
-         .B16(2)
-         .B16(16)
-         .append_repeated(0, 4)
-         .B32(48000 << 16)
-         .append_bytes(ac3.as_slice())
-    });
-
-    let mut iter = super::BoxIter::new(&mut stream);
-    let mut stream = iter.next_box().unwrap().unwrap();
-    let (codec_type, _) = super::read_audio_sample_entry(&mut stream)
-          .expect("fail to read ac3 atom");
-    assert_eq!(codec_type, super::CodecType::AC3);
-}
-#[test]
 fn read_stsd_mp4v() {
     let mp4v =
         vec![
                                                 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
             0x00, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00,
             0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xd0, 0x01, 0xe0, 0x00, 0x48,
             0x00, 0x00, 0x00, 0x48, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01,
             0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20, 0x20,
@@ -1026,58 +1001,23 @@ fn max_table_limit() {
     }).into_inner();
     let mut stream = make_box(BoxSize::Auto, b"edts", |s| {
         s.append_bytes(elst.as_slice())
     });
     let mut iter = super::BoxIter::new(&mut stream);
     let mut stream = iter.next_box().unwrap().unwrap();
     let mut track = super::Track::new(0);
     match super::read_edts(&mut stream, &mut track) {
-        Err(Error::Unsupported(s)) => assert_eq!(s, "Over limited value"),
+        Err(Error::TableTooLarge) => (),
         Ok(_) => panic!("expected an error result"),
         _ => panic!("expected a different error result"),
     }
 }
 
 #[test]
-fn jpeg_video_sample_entry() {
-    let jpeg = make_box(BoxSize::Auto, b"jpeg", |s| {
-        s.append_repeated(0, 6)
-         .B16(1)
-         .append_repeated(0, 16)
-         .B16(1024)
-         .B16(1024)
-         .append_repeated(0, 14)
-         .append_repeated(0, 32)
-         .append_repeated(0, 4)
-    }).into_inner();
-    let mut stream = make_fullbox(BoxSize::Auto, b"stsd", 0, |s| {
-        s.B32(1)
-         .append_bytes(jpeg.as_slice())
-    });
-
-    let mut iter = super::BoxIter::new(&mut stream);
-    let mut stream = iter.next_box().unwrap().unwrap();
-    let mut track = super::Track::new(0);
-    match super::read_stsd(&mut stream, &mut track) {
-        Ok(sample_description) => {
-            match sample_description.descriptions[0] {
-                super::SampleEntry::Video(ref jpeg) => {
-                    assert_eq!(track.codec_type, super::CodecType::JPEG);
-                    assert_eq!(jpeg.height, 1024);
-                    assert_eq!(jpeg.width, 1024);
-                } ,
-                _ => {},
-            }
-        },
-        _ => panic!("failed to parse a jpeg atom"),
-    }
-}
-
-#[test]
 fn unknown_video_sample_entry() {
     let unknown_codec = make_box(BoxSize::Auto, b"yyyy", |s| {
         s.append_repeated(0, 16)
     }).into_inner();
     let mut stream = make_box(BoxSize::Auto, b"xxxx", |s| {
         s.append_repeated(0, 6)
          .B16(1)
          .append_repeated(0, 16)
--- a/media/libstagefright/binding/mp4parse/tests/public.rs
+++ b/media/libstagefright/binding/mp4parse/tests/public.rs
@@ -55,19 +55,16 @@ fn public_api() {
                         assert!(vpx.chroma_subsampling > 0);
                         assert!(!vpx.codec_init.is_empty());
                         "VPx"
                     }
                     mp4::VideoCodecSpecific::ESDSConfig(mp4v) => {
                         assert!(!mp4v.is_empty());
                         "MP4V"
                     }
-                    mp4::VideoCodecSpecific::JPEG => {
-                        "JPEG"
-                    }
                 }, "AVC");
             }
             Some(mp4::SampleEntry::Audio(a)) => {
                 // track part
                 assert_eq!(track.duration, Some(mp4::TrackScaledTime(2944, 1)));
                 assert_eq!(track.empty_duration, Some(mp4::MediaScaledTime(0)));
                 assert_eq!(track.media_time, Some(mp4::TrackScaledTime(1024, 1)));
                 assert_eq!(track.timescale, Some(mp4::TrackTimeScale(48000, 1)));
@@ -97,22 +94,16 @@ fn public_api() {
                     mp4::AudioCodecSpecific::OpusSpecificBox(opus) => {
                         // We don't enter in here, we just check if fields are public.
                         assert!(opus.version > 0);
                         "Opus"
                     }
                     mp4::AudioCodecSpecific::MP3 => {
                         "MP3"
                     }
-                    mp4::AudioCodecSpecific::AC3SpecificBox => {
-                        "AC3"
-                    }
-                    mp4::AudioCodecSpecific::EC3SpecificBox => {
-                        "EC3"
-                    }
                 }, "ES");
                 assert!(a.samplesize > 0);
                 assert!(a.samplerate > 0);
             }
             Some(mp4::SampleEntry::Unknown) | None => {}
         }
     }
 }
--- a/media/libstagefright/binding/mp4parse_capi/Cargo.toml
+++ b/media/libstagefright/binding/mp4parse_capi/Cargo.toml
@@ -3,28 +3,25 @@ name = "mp4parse_capi"
 version = "0.8.0"
 authors = [
   "Ralph Giles <giles@mozilla.com>",
   "Matthew Gregan <kinetik@flim.org>",
   "Alfredo Yang <ayang@mozilla.com>",
 ]
 
 description = "Parser for ISO base media file format (mp4)"
-documentation = "https://mp4parse-docs.surge.sh/mp4parse/"
+documentation = "https://docs.rs/mp4parse_capi/"
 license = "MPL-2.0"
 
 repository = "https://github.com/mozilla/mp4parse-rust"
 
 # Avoid complaints about trying to package test files.
 exclude = [
   "*.mp4",
 ]
 
 build = false
 
 [dependencies]
 byteorder = "1.0.0"
 mp4parse = {version = "0.8.0", path = "../mp4parse"}
 num-traits = "0.1.37"
 
-# Somewhat heavy-handed, but we want at least -Z force-overflow-checks=on.
-[profile.release]
-debug-assertions = true
--- a/media/libstagefright/binding/mp4parse_capi/src/lib.rs
+++ b/media/libstagefright/binding/mp4parse_capi/src/lib.rs
@@ -64,16 +64,17 @@ use mp4parse::Track;
 #[derive(PartialEq, Debug)]
 pub enum mp4parse_status {
     OK = 0,
     BAD_ARG = 1,
     INVALID = 2,
     UNSUPPORTED = 3,
     EOF = 4,
     IO = 5,
+    TABLE_TOO_LARGE = 6,
 }
 
 #[allow(non_camel_case_types)]
 #[repr(C)]
 #[derive(PartialEq, Debug)]
 pub enum mp4parse_track_type {
     VIDEO = 0,
     AUDIO = 1,
@@ -330,17 +331,18 @@ pub unsafe extern fn mp4parse_read(parse
         Err(Error::UnexpectedEOF) => mp4parse_status::EOF,
         Err(Error::Io(_)) => {
             // Block further calls after a read failure.
             // Getting std::io::ErrorKind::UnexpectedEof is normal
             // but our From trait implementation should have converted
             // those to our Error::UnexpectedEOF variant.
             (*parser).set_poisoned(true);
             mp4parse_status::IO
-        }
+        },
+        Err(Error::TableTooLarge) => mp4parse_status::TABLE_TOO_LARGE,
     }
 }
 
 /// Return the number of tracks parsed by previous `mp4parse_read()` call.
 #[no_mangle]
 pub unsafe extern fn mp4parse_get_track_count(parser: *const mp4parse_parser, count: *mut u32) -> mp4parse_status {
     // Validate arguments from C.
     if parser.is_null() || count.is_null() || (*parser).poisoned() {
@@ -412,44 +414,39 @@ pub unsafe extern fn mp4parse_get_track_
     }
 
     info.track_type = match context.tracks[track_index].track_type {
         TrackType::Video => mp4parse_track_type::VIDEO,
         TrackType::Audio => mp4parse_track_type::AUDIO,
         TrackType::Unknown => return mp4parse_status::UNSUPPORTED,
     };
 
+    // Return UNKNOWN for unsupported format.
     info.codec = match context.tracks[track_index].data {
         Some(SampleEntry::Audio(ref audio)) => match audio.codec_specific {
             AudioCodecSpecific::OpusSpecificBox(_) =>
                 mp4parse_codec::OPUS,
             AudioCodecSpecific::FLACSpecificBox(_) =>
                 mp4parse_codec::FLAC,
             AudioCodecSpecific::ES_Descriptor(ref esds) if esds.audio_codec == CodecType::AAC =>
                 mp4parse_codec::AAC,
             AudioCodecSpecific::ES_Descriptor(ref esds) if esds.audio_codec == CodecType::MP3 =>
                 mp4parse_codec::MP3,
             AudioCodecSpecific::ES_Descriptor(_) =>
                 mp4parse_codec::UNKNOWN,
             AudioCodecSpecific::MP3 =>
                 mp4parse_codec::MP3,
-            AudioCodecSpecific::AC3SpecificBox =>
-                mp4parse_codec::AC3,
-            AudioCodecSpecific::EC3SpecificBox =>
-                mp4parse_codec::EC3,
         },
         Some(SampleEntry::Video(ref video)) => match video.codec_specific {
             VideoCodecSpecific::VPxConfig(_) =>
                 mp4parse_codec::VP9,
             VideoCodecSpecific::AVCConfig(_) =>
                 mp4parse_codec::AVC,
-            VideoCodecSpecific::ESDSConfig(_) =>
-                mp4parse_codec::MP4V,
-            VideoCodecSpecific::JPEG =>
-                mp4parse_codec::JPEG,
+            VideoCodecSpecific::ESDSConfig(_) => // MP4V (14496-2) video is unsupported.
+                mp4parse_codec::UNKNOWN,
         },
         _ => mp4parse_codec::UNKNOWN,
     };
 
     let track = &context.tracks[track_index];
 
     if let (Some(track_timescale),
             Some(context_timescale)) = (track.timescale,
@@ -568,18 +565,16 @@ pub unsafe extern fn mp4parse_get_track_
                             return mp4parse_status::INVALID;
                         }
                         (*info).extra_data.length = v.len() as u32;
                         (*info).extra_data.data = v.as_ptr();
                     }
                 }
             }
         }
-        AudioCodecSpecific::AC3SpecificBox => (),
-        AudioCodecSpecific::EC3SpecificBox => (),
         AudioCodecSpecific::MP3 => (),
     }
 
     if let Some(p) = audio.protection_info.iter().find(|sinf| sinf.tenc.is_some()) {
         if let Some(ref tenc) = p.tenc {
             (*info).protected_data.is_encrypted = tenc.is_encrypted;
             (*info).protected_data.iv_size = tenc.iv_size;
             (*info).protected_data.kid.set_data(&(tenc.kid));
--- a/media/libstagefright/binding/update-rust.sh
+++ b/media/libstagefright/binding/update-rust.sh
@@ -1,13 +1,13 @@
 #!/bin/sh -e
 # Script to update mp4parse-rust sources to latest upstream
 
 # Default version.
-VER=ce7e2e66613009d56aea2588c4d11a23d4dd8056
+VER=33e7ea77aaf06939b4aaf284f487cdd13d3db671
 
 # Accept version or commit from the command line.
 if test -n "$1"; then
   VER=$1
 fi
 
 echo "Fetching sources..."
 rm -rf _upstream
--- a/media/libstagefright/gtest/TestParser.cpp
+++ b/media/libstagefright/gtest/TestParser.cpp
@@ -216,17 +216,17 @@ static const TestFileData testFiles[] = 
 };
 
 static const TestFileData rustTestFiles[] = {
   // filename                      #V dur   w    h  #A dur  crypt        off   moof  headr  audio_profile
   { "test_case_1156505.mp4",        0, -1,   0,   0, 0, -1, false, 152, false, false, 0 },
   { "test_case_1181213.mp4",        1, 416666,
                                            320, 240, 1, 477460,
                                                              true,   0, false, false, 2 },
-  { "test_case_1181215.mp4",        1, 4966666,   190,   240, 0, -1, false,   0, false, false, 0 },
+  { "test_case_1181215.mp4",        0, -1,   0,   0, 0, -1, false,   0, false, false, 0 },
   { "test_case_1181220.mp4",        0, -1,   0,   0, 0, -1, false,   0, false, false, 0 },
   { "test_case_1181223.mp4",        0, 416666,
                                            320, 240, 0, -1, false,   0, false, false, 0 },
   { "test_case_1181719.mp4",        0, -1,   0,   0, 0, -1, false,   0, false, false, 0 },
   { "test_case_1185230.mp4",        2, 416666,
                                            320, 240, 2,  5, false,   0, false, false, 2 },
   { "test_case_1187067.mp4",        1, 80000,
                                            160,  90, 0, -1, false,   0, false, false, 0 },
@@ -301,73 +301,70 @@ TEST(stagefright_MPEG4Metadata, test_cas
         << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
       EXPECT_EQ(None, metadata.GetNumberTracks(TrackInfo::kTextTrack).Ref())
         << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
       EXPECT_EQ(None, metadata.GetNumberTracks(static_cast<TrackInfo::TrackType>(-1)).Ref())
         << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
       EXPECT_FALSE(metadata.GetTrackInfo(TrackInfo::kUndefinedTrack, 0).Ref());
       MP4Metadata::ResultAndTrackInfo trackInfo =
         metadata.GetTrackInfo(TrackInfo::kVideoTrack, 0);
-      if (tests[test].mNumberVideoTracks == 0 ||
-          tests[test].mNumberVideoTracks == E) {
-        EXPECT_TRUE(!trackInfo.Ref());
-      } else {
+      if (!!tests[test].mNumberVideoTracks) {
         ASSERT_TRUE(!!trackInfo.Ref());
         const VideoInfo* videoInfo = trackInfo.Ref()->GetAsVideoInfo();
         ASSERT_TRUE(!!videoInfo);
-        EXPECT_TRUE(videoInfo->IsValid());
-        EXPECT_TRUE(videoInfo->IsVideo());
+        EXPECT_TRUE(videoInfo->IsValid()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
+        EXPECT_TRUE(videoInfo->IsVideo()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
         EXPECT_EQ(tests[test].mVideoDuration,
-                  videoInfo->mDuration.ToMicroseconds());
-        EXPECT_EQ(tests[test].mWidth, videoInfo->mDisplay.width);
-        EXPECT_EQ(tests[test].mHeight, videoInfo->mDisplay.height);
+                  videoInfo->mDuration.ToMicroseconds()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
+        EXPECT_EQ(tests[test].mWidth, videoInfo->mDisplay.width) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
+        EXPECT_EQ(tests[test].mHeight, videoInfo->mDisplay.height) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
 
         MP4Metadata::ResultAndIndice indices =
           metadata.GetTrackIndice(videoInfo->mTrackId);
-        EXPECT_TRUE(!!indices.Ref());
+        EXPECT_TRUE(!!indices.Ref()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
         for (size_t i = 0; i < indices.Ref()->Length(); i++) {
           Index::Indice data;
-          EXPECT_TRUE(indices.Ref()->GetIndice(i, data));
-          EXPECT_TRUE(data.start_offset <= data.end_offset);
-          EXPECT_TRUE(data.start_composition <= data.end_composition);
+          EXPECT_TRUE(indices.Ref()->GetIndice(i, data)) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
+          EXPECT_TRUE(data.start_offset <= data.end_offset) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
+          EXPECT_TRUE(data.start_composition <= data.end_composition) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
         }
       }
       trackInfo = metadata.GetTrackInfo(TrackInfo::kAudioTrack, 0);
       if (tests[test].mNumberAudioTracks == 0 ||
           tests[test].mNumberAudioTracks == E) {
-        EXPECT_TRUE(!trackInfo.Ref());
+        EXPECT_TRUE(!trackInfo.Ref()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
       } else {
         ASSERT_TRUE(!!trackInfo.Ref());
         const AudioInfo* audioInfo = trackInfo.Ref()->GetAsAudioInfo();
         ASSERT_TRUE(!!audioInfo);
-        EXPECT_TRUE(audioInfo->IsValid());
-        EXPECT_TRUE(audioInfo->IsAudio());
+        EXPECT_TRUE(audioInfo->IsValid()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
+        EXPECT_TRUE(audioInfo->IsAudio()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
         EXPECT_EQ(tests[test].mAudioDuration,
-                  audioInfo->mDuration.ToMicroseconds());
-        EXPECT_EQ(tests[test].mAudioProfile, audioInfo->mProfile);
+                  audioInfo->mDuration.ToMicroseconds()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
+        EXPECT_EQ(tests[test].mAudioProfile, audioInfo->mProfile) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
         if (tests[test].mAudioDuration !=
             audioInfo->mDuration.ToMicroseconds()) {
           MOZ_RELEASE_ASSERT(false);
         }
 
         MP4Metadata::ResultAndIndice indices =
           metadata.GetTrackIndice(audioInfo->mTrackId);
-        EXPECT_TRUE(!!indices.Ref());
+        EXPECT_TRUE(!!indices.Ref()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
         for (size_t i = 0; i < indices.Ref()->Length(); i++) {
           Index::Indice data;
-          EXPECT_TRUE(indices.Ref()->GetIndice(i, data));
-          EXPECT_TRUE(data.start_offset <= data.end_offset);
-          EXPECT_TRUE(int64_t(data.start_composition) <= int64_t(data.end_composition));
+          EXPECT_TRUE(indices.Ref()->GetIndice(i, data)) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
+          EXPECT_TRUE(data.start_offset <= data.end_offset) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
+          EXPECT_TRUE(int64_t(data.start_composition) <= int64_t(data.end_composition)) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
         }
       }
-      EXPECT_FALSE(metadata.GetTrackInfo(TrackInfo::kTextTrack, 0).Ref());
-      EXPECT_FALSE(metadata.GetTrackInfo(static_cast<TrackInfo::TrackType>(-1), 0).Ref());
+      EXPECT_FALSE(metadata.GetTrackInfo(TrackInfo::kTextTrack, 0).Ref()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
+      EXPECT_FALSE(metadata.GetTrackInfo(static_cast<TrackInfo::TrackType>(-1), 0).Ref()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
       // We can see anywhere in any MPEG4.
-      EXPECT_TRUE(metadata.CanSeek());
-      EXPECT_EQ(tests[test].mHasCrypto, metadata.Crypto().Ref()->valid);
+      EXPECT_TRUE(metadata.CanSeek()) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
+      EXPECT_EQ(tests[test].mHasCrypto, metadata.Crypto().Ref()->valid) << (rust ? "rust/" : "stagefright/") << tests[test].mFilename;
     }
   }
 }
 
 // Bug 1224019: This test produces way to much output, disabling for now.
 #if 0
 TEST(stagefright_MPEG4Metadata, test_case_mp4_subsets)
 {
--- a/media/webrtc/signaling/signaling.gyp
+++ b/media/webrtc/signaling/signaling.gyp
@@ -69,16 +69,17 @@
         './src/common/time_profiling',
         './src/media',
         './src/media-conduit',
         './src/mediapipeline',
         './src/peerconnection',
         './src/sdp/sipcc',
         '../../../dom/base',
         '../../../dom/media',
+        '../../../dom/media/platforms',
         '../../../media/mtransport',
         '../trunk',
         '../../libyuv/libyuv/include',
         '../../mtransport/third_party/nrappkit/src/util/libekr',
       ],
 
       #
       # DEPENDENCIES
@@ -219,37 +220,16 @@
           'include_dirs!': [
             '../trunk/webrtc',
           ],
           'include_dirs': [
             '../../../netwerk/srtp/src/include',
             '../../../netwerk/srtp/src/crypto/include',
           ],
         }],
-        ['moz_webrtc_omx==1', {
-          'sources': [
-            './src/media-conduit/WebrtcOMXH264VideoCodec.cpp',
-            './src/media-conduit/OMXVideoCodec.cpp',
-          ],
-          'include_dirs': [
-            # hack on hack to re-add it after SrtpFlow removes it
-            '../../../dom/media/omx',
-            '../../../gfx/layers/client',
-          ],
-          'cflags_mozilla': [
-            '-I$(ANDROID_SOURCE)/frameworks/av/include/media/stagefright',
-            '-I$(ANDROID_SOURCE)/frameworks/av/include',
-            '-I$(ANDROID_SOURCE)/frameworks/native/include/media/openmax',
-            '-I$(ANDROID_SOURCE)/frameworks/native/include',
-            '-I$(ANDROID_SOURCE)/frameworks/native/opengl/include',
-          ],
-          'defines' : [
-            'MOZ_WEBRTC_OMX'
-          ],
-        }],
         ['moz_webrtc_mediacodec==1', {
           'include_dirs': [
             '../../../widget/android',
           ],
           'sources': [
             './src/media-conduit/MediaCodecVideoCodec.h',
             './src/media-conduit/WebrtcMediaCodecVP8VideoCodec.h',
             './src/media-conduit/MediaCodecVideoCodec.cpp',
@@ -274,17 +254,19 @@
           'defines' : [
             'NO_CHROMIUM_LOGGING',
             'USE_FAKE_PCOBSERVER',
           ],
         }],
         ['build_for_standalone==0', {
           'sources': [
             './src/media-conduit/GmpVideoCodec.cpp',
+            './src/media-conduit/MediaDataDecoderCodec.cpp',
             './src/media-conduit/WebrtcGmpVideoCodec.cpp',
+            './src/media-conduit/WebrtcMediaDataDecoderCodec.cpp',
           ],
         }],
         ['build_for_standalone!=0', {
           'include_dirs': [
             './test'
           ],
           'defines' : [
             'NO_CHROMIUM_LOGGING',
--- a/media/webrtc/signaling/src/media-conduit/MediaConduitInterface.h
+++ b/media/webrtc/signaling/src/media-conduit/MediaConduitInterface.h
@@ -101,31 +101,16 @@ public:
    * @param len  : Length of the RTCP packet
    * @result     : NS_OK on success, NS_ERROR_FAILURE otherwise
    */
   virtual nsresult SendRtcpPacket(const uint8_t* data, size_t len) = 0;
   NS_INLINE_DECL_THREADSAFE_REFCOUNTING(TransportInterface)
 };
 
 /**
- * This class wraps image object for VideoRenderer::RenderVideoFrame()
- * callback implementation to use for rendering.
- */
-class ImageHandle
-{
-public:
-  explicit ImageHandle(layers::Image* image) : mImage(image) {}
-
-  const RefPtr<layers::Image>& GetImage() const { return mImage; }
-
-private:
-  RefPtr<layers::Image> mImage;
-};
-
-/**
  * 1. Abstract renderer for video data
  * 2. This class acts as abstract interface between the video-engine and
  *    video-engine agnostic renderer implementation.
  * 3. Concrete implementation of this interface is responsible for
  *    processing and/or rendering the obtained raw video frame to appropriate
  *    output , say, <video>
  */
 class VideoRenderer
@@ -140,45 +125,31 @@ public:
    * @param height: current height of the video @ decoder
    * @param number_of_streams: number of participating video streams
    */
   virtual void FrameSizeChange(unsigned int width,
                                unsigned int height,
                                unsigned int number_of_streams) = 0;
 
   /**
-   * Callback Function reporting decoded I420 frame for processing.
-   * @param buffer: pointer to decoded video frame
+   * Callback Function reporting decoded frame for processing.
+   * @param buffer: reference to decoded video frame
    * @param buffer_size: size of the decoded frame
    * @param time_stamp: Decoder timestamp, typically 90KHz as per RTP
    * @render_time: Wall-clock time at the decoder for synchronization
    *                purposes in milliseconds
-   * @handle: opaque handle for image object of decoded video frame.
    * NOTE: If decoded video frame is passed through buffer , it is the
    * responsibility of the concrete implementations of this class to own copy
    * of the frame if needed for time longer than scope of this callback.
    * Such implementations should be quick in processing the frames and return
    * immediately.
-   * On the other hand, if decoded video frame is passed through handle, the
-   * implementations should keep a reference to the (ref-counted) image object
-   * inside until it's no longer needed.
    */
   virtual void RenderVideoFrame(const webrtc::VideoFrameBuffer& buffer,
                                 uint32_t time_stamp,
-                                int64_t render_time,
-                                const ImageHandle& handle) = 0;
-  virtual void RenderVideoFrame(const uint8_t* buffer_y,
-                                uint32_t y_stride,
-                                const uint8_t* buffer_u,
-                                uint32_t u_stride,
-                                const uint8_t* buffer_v,
-                                uint32_t v_stride,
-                                uint32_t time_stamp,
-                                int64_t render_time,
-                                const ImageHandle& handle) = 0;
+                                int64_t render_time) = 0;
 
   NS_INLINE_DECL_THREADSAFE_REFCOUNTING(VideoRenderer)
 };
 
 
 /**
  * Generic Interface for representing Audio/Video Session
  * MediaSession conduit is identified by 2 main components
new file mode 100644
--- /dev/null
+++ b/media/webrtc/signaling/src/media-conduit/MediaDataDecoderCodec.cpp
@@ -0,0 +1,37 @@
+/* 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/. */
+
+#include "MediaDataDecoderCodec.h"
+#include "MediaPrefs.h"
+#include "WebrtcMediaDataDecoderCodec.h"
+
+namespace mozilla {
+
+/* static */ WebrtcVideoEncoder*
+MediaDataDecoderCodec::CreateEncoder(
+  webrtc::VideoCodecType aCodecType)
+{
+  return nullptr;
+}
+
+/* static */ WebrtcVideoDecoder*
+MediaDataDecoderCodec::CreateDecoder(
+  webrtc::VideoCodecType aCodecType)
+{
+  if (!MediaPrefs::MediaDataDecoderEnabled()) {
+    return nullptr;
+  }
+
+  switch (aCodecType) {
+    case webrtc::VideoCodecType::kVideoCodecVP8:
+    case webrtc::VideoCodecType::kVideoCodecVP9:
+    case webrtc::VideoCodecType::kVideoCodecH264:
+      break;
+    default:
+      return nullptr;
+  }
+  return new WebrtcMediaDataDecoder();
+}
+
+} // namespace mozilla
\ No newline at end of file
new file mode 100644
--- /dev/null
+++ b/media/webrtc/signaling/src/media-conduit/MediaDataDecoderCodec.h
@@ -0,0 +1,34 @@
+/* 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/. */
+
+#ifndef MEDIA_DATA_DECODER_CODEC_H_
+#define MEDIA_DATA_DECODER_CODEC_H_
+
+#include "MediaConduitInterface.h"
+#include "webrtc/common_types.h"
+
+namespace mozilla {
+
+class WebrtcVideoDecoder;
+class WebrtcVideoEncoder;
+class MediaDataDecoderCodec
+{
+ public:
+  /**
+   * Create encoder object for codec type |aCodecType|. Return |nullptr| when
+   * failed.
+   */
+  static WebrtcVideoEncoder* CreateEncoder(
+    webrtc::VideoCodecType aCodecType);
+
+  /**
+   * Create decoder object for codec type |aCodecType|. Return |nullptr| when
+   * failed.
+   */
+  static WebrtcVideoDecoder* CreateDecoder(
+    webrtc::VideoCodecType aCodecType);
+};
+}
+
+#endif // MEDIA_DATA_DECODER_CODEC_H_
deleted file mode 100644
--- a/media/webrtc/signaling/src/media-conduit/OMXVideoCodec.cpp
+++ /dev/null
@@ -1,30 +0,0 @@
-/* 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/. */
-
-#include "OMXVideoCodec.h"
-
-#ifdef WEBRTC_GONK
-#include "WebrtcOMXH264VideoCodec.h"
-#endif
-
-namespace mozilla {
-
-WebrtcVideoEncoder*
-OMXVideoCodec::CreateEncoder(CodecType aCodecType)
-{
-  if (aCodecType == CODEC_H264) {
-    return new WebrtcOMXH264VideoEncoder();
-  }
-  return nullptr;
-}
-
-WebrtcVideoDecoder*
-OMXVideoCodec::CreateDecoder(CodecType aCodecType) {
-  if (aCodecType == CODEC_H264) {
-    return new WebrtcOMXH264VideoDecoder();
-  }
-  return nullptr;
-}
-
-}
deleted file mode 100644
--- a/media/webrtc/signaling/src/media-conduit/OMXVideoCodec.h
+++ /dev/null
@@ -1,32 +0,0 @@
-/* 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/. */
-
-#ifndef OMX_VIDEO_CODEC_H_
-#define OMX_VIDEO_CODEC_H_
-
-#include "MediaConduitInterface.h"
-
-namespace mozilla {
-class OMXVideoCodec {
- public:
-  enum CodecType {
-    CODEC_H264,
-  };
-
-  /**
-   * Create encoder object for codec type |aCodecType|. Return |nullptr| when
-   * failed.
-   */
-  static WebrtcVideoEncoder* CreateEncoder(CodecType aCodecType);
-
-  /**
-   * Create decoder object for codec type |aCodecType|. Return |nullptr| when
-   * failed.
-   */
-  static WebrtcVideoDecoder* CreateDecoder(CodecType aCodecType);
-};
-
-}
-
-#endif // OMX_VIDEO_CODEC_H_
--- a/media/webrtc/signaling/src/media-conduit/VideoConduit.cpp
+++ b/media/webrtc/signaling/src/media-conduit/VideoConduit.cpp
@@ -47,16 +47,18 @@
 #include "OMXVideoCodec.h"
 #endif
 
 #ifdef MOZ_WEBRTC_MEDIACODEC
 #include "MediaCodecVideoCodec.h"
 #endif
 #include "WebrtcGmpVideoCodec.h"
 
+#include "MediaDataDecoderCodec.h"
+
 // for ntohs
 #ifdef _MSC_VER
 #include "Winsock2.h"
 #else
 #include <netinet/in.h>
 #endif
 
 #include <algorithm>
@@ -1432,16 +1434,22 @@ WebrtcVideoConduit::ConfigureRecvMediaCo
 webrtc::VideoDecoder*
 WebrtcVideoConduit::CreateDecoder(webrtc::VideoCodecType aType)
 {
   webrtc::VideoDecoder* decoder = nullptr;
 #ifdef MOZ_WEBRTC_MEDIACODEC
   bool enabled = false;
 #endif
 
+  // Attempt to create a decoder using MediaDataDecoder.
+  decoder = MediaDataDecoderCodec::CreateDecoder(aType);
+  if (decoder) {
+    return decoder;
+  }
+
   switch (aType) {
     case webrtc::VideoCodecType::kVideoCodecH264:
       // get an external decoder
 #ifdef MOZ_WEBRTC_OMX
       decoder = OMXVideoCodec::CreateDecoder(OMXVideoCodec::CodecType::CODEC_H264);
 #else
       decoder = GmpVideoCodec::CreateDecoder();
 #endif
@@ -2305,21 +2313,19 @@ WebrtcVideoConduit::OnFrame(const webrtc
                                  const_cast<unsigned char*>(video_frame.video_frame_buffer()->DataY()),
                                  reinterpret_cast<unsigned char*>(&timestamp),
                                  sizeof(timestamp), 0, 0);
     if (ok) {
       VideoLatencyUpdate(now - timestamp);
     }
   }
 
-  const ImageHandle img_handle(nullptr);
   mRenderer->RenderVideoFrame(*video_frame.video_frame_buffer(),
                               video_frame.timestamp(),
-                              video_frame.render_time_ms(),
-                              img_handle);
+                              video_frame.render_time_ms());
 }
 
 // Compare lists of codecs
 bool
 WebrtcVideoConduit::CodecsDifferent(const nsTArray<UniquePtr<VideoCodecConfig>>& a,
                                     const nsTArray<UniquePtr<VideoCodecConfig>>& b)
 {
   // return a != b;
new file mode 100644
--- /dev/null
+++ b/media/webrtc/signaling/src/media-conduit/WebrtcMediaDataDecoderCodec.cpp
@@ -0,0 +1,255 @@
+/* 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/. */
+
+#include "WebrtcMediaDataDecoderCodec.h"
+#include "ImageContainer.h"
+#include "Layers.h"
+#include "PDMFactory.h"
+#include "VideoUtils.h"
+#include "mozilla/layers/ImageBridgeChild.h"
+#include "webrtc/base/keep_ref_until_done.h"
+
+namespace mozilla {
+
+WebrtcMediaDataDecoder::WebrtcMediaDataDecoder()
+  : mTaskQueue(
+      new TaskQueue(GetMediaThreadPool(MediaThreadType::PLATFORM_DECODER),
+                    "WebrtcMediaDataDecoder::mTaskQueue"))
+  , mImageContainer(layers::LayerManager::CreateImageContainer(
+      layers::ImageContainer::ASYNCHRONOUS))
+  , mFactory(new PDMFactory())
+  , mMonitor("WebrtcMediaDataDecoder")
+{
+}
+
+WebrtcMediaDataDecoder::~WebrtcMediaDataDecoder()
+{
+  mTaskQueue->BeginShutdown();
+  mTaskQueue->AwaitShutdownAndIdle();
+}
+
+int32_t
+WebrtcMediaDataDecoder::InitDecode(const webrtc::VideoCodec* aCodecSettings,
+                                   int32_t aNumberOfCores)
+{
+  nsCString codec;
+  switch (aCodecSettings->codecType) {
+    case webrtc::VideoCodecType::kVideoCodecVP8:
+      codec = "video/webm; codecs=vp8";
+      break;
+    case webrtc::VideoCodecType::kVideoCodecVP9:
+      codec = "video/webm; codecs=vp9";
+      break;
+    case webrtc::VideoCodecType::kVideoCodecH264:
+      codec = "video/avc";
+      break;
+    default:
+      return WEBRTC_VIDEO_CODEC_ERROR;
+  }
+
+  mTrackType = TrackInfo::kVideoTrack;
+
+  mInfo = VideoInfo(aCodecSettings->width, aCodecSettings->height);
+  mInfo.mMimeType = codec;
+
+  RefPtr<layers::KnowsCompositor> knowsCompositor =
+    layers::ImageBridgeChild::GetSingleton();
+
+  mDecoder = mFactory->CreateDecoder(
+    { mInfo,
+      mTaskQueue,
+      CreateDecoderParams::OptionSet(CreateDecoderParams::Option::LowLatency),
+      mTrackType,
+      mImageContainer,
+      knowsCompositor });
+
+  if (!mDecoder) {
+    return WEBRTC_VIDEO_CODEC_ERROR;
+  }
+
+  MonitorAutoLock lock(mMonitor);
+  bool done = false;
+  mDecoder->Init()->Then(mTaskQueue,
+                         __func__,
+                         [&](TrackInfo::TrackType) {
+                           MonitorAutoLock lock(mMonitor);
+                           done = true;
+                           mMonitor.Notify();
+                         },
+                         [&](const MediaResult& aError) {
+                           MonitorAutoLock lock(mMonitor);
+                           done = true;
+                           mError = aError;
+                           mMonitor.Notify();
+                         });
+
+  while (!done) {
+    mMonitor.Wait();
+  }
+
+  return NS_SUCCEEDED(mError) ? WEBRTC_VIDEO_CODEC_OK : WEBRTC_VIDEO_CODEC_ERROR;
+}
+
+int32_t
+WebrtcMediaDataDecoder::Decode(
+  const webrtc::EncodedImage& aInputImage,
+  bool aMissingFrames,
+  const webrtc::RTPFragmentationHeader* aFragmentation,
+  const webrtc::CodecSpecificInfo* aCodecSpecificInfo,
+  int64_t aRenderTimeMs)
+{
+  if (!mCallback || !mDecoder) {
+    return WEBRTC_VIDEO_CODEC_UNINITIALIZED;
+  }
+
+  if (!aInputImage._buffer || !aInputImage._length) {
+    return WEBRTC_VIDEO_CODEC_ERR_PARAMETER;
+  }
+
+  // Always start with a complete key frame.
+  if (mNeedKeyframe) {
+    if (aInputImage._frameType != webrtc::FrameType::kVideoFrameKey)
+      return WEBRTC_VIDEO_CODEC_ERROR;
+    // We have a key frame - is it complete?
+    if (aInputImage._completeFrame) {
+      mNeedKeyframe = false;
+    } else {
+      return WEBRTC_VIDEO_CODEC_ERROR;
+    }
+  }
+
+  RefPtr<MediaRawData> compressedFrame =
+    new MediaRawData(aInputImage._buffer, aInputImage._length);
+  if (!compressedFrame->Data()) {
+    return WEBRTC_VIDEO_CODEC_MEMORY;
+  }
+
+  compressedFrame->mTime =
+    media::TimeUnit::FromMicroseconds(aInputImage._timeStamp);
+  compressedFrame->mTimecode =
+    media::TimeUnit::FromMicroseconds(aRenderTimeMs * 1000);
+  compressedFrame->mKeyframe =
+    aInputImage._frameType == webrtc::FrameType::kVideoFrameKey;
+  {
+    MonitorAutoLock lock(mMonitor);
+    bool done = false;
+    mDecoder->Decode(compressedFrame)->Then(
+      mTaskQueue,
+      __func__,
+      [&](const MediaDataDecoder::DecodedData& aResults) {
+        MonitorAutoLock lock(mMonitor);
+        mResults = aResults;
+        done = true;
+        mMonitor.Notify();
+      },
+      [&](const MediaResult& aError) {
+        MonitorAutoLock lock(mMonitor);
+        mError = aError;
+        done = true;
+        mMonitor.Notify();
+      });
+
+    while (!done) {
+      mMonitor.Wait();
+    }
+
+    for (auto& frame : mResults) {
+      MOZ_ASSERT(frame->mType == MediaData::VIDEO_DATA);
+      RefPtr<VideoData> video = frame->As<VideoData>();
+      MOZ_ASSERT(video);
+      if (!video->mImage) {
+        // Nothing to display.
+        continue;
+      }
+      rtc::scoped_refptr<ImageBuffer> image(
+        new rtc::RefCountedObject<ImageBuffer>(Move(video->mImage)));
+
+      webrtc::VideoFrame videoFrame(image,
+                                    frame->mTime.ToMicroseconds(),
+                                    frame->mDuration.ToMicroseconds() * 1000,
+                                    aInputImage.rotation_);
+      mCallback->Decoded(videoFrame);
+    }
+    mResults.Clear();
+  }
+  return NS_SUCCEEDED(mError) ? WEBRTC_VIDEO_CODEC_OK
+                              : WEBRTC_VIDEO_CODEC_ERROR;
+}
+
+int32_t
+WebrtcMediaDataDecoder::RegisterDecodeCompleteCallback(
+  webrtc::DecodedImageCallback* aCallback)
+{
+  mCallback = aCallback;
+  return WEBRTC_VIDEO_CODEC_OK;
+}
+
+int32_t
+WebrtcMediaDataDecoder::Release()
+{
+  MonitorAutoLock lock(mMonitor);
+  bool done = false;
+  mDecoder->Flush()
+    ->Then(mTaskQueue,
+           __func__,
+           [this]() { return mDecoder->Shutdown(); },
+           [this](const MediaResult& aError) { return mDecoder->Shutdown(); })
+    ->Then(mTaskQueue,
+           __func__,
+           [&]() {
+             MonitorAutoLock lock(mMonitor);
+             done = true;
+             mMonitor.Notify();
+           },
+           []() { MOZ_ASSERT_UNREACHABLE("Shutdown promise always resolved"); });
+
+  while (!done) {
+    mMonitor.Wait();
+  }
+
+  mDecoder = nullptr;
+  mNeedKeyframe = true;
+
+  return WEBRTC_VIDEO_CODEC_OK;
+}
+
+bool
+WebrtcMediaDataDecoder::OnTaskQueue() const
+{
+  return OwnerThread()->IsCurrentThreadIn();
+}
+
+ImageBuffer::ImageBuffer(RefPtr<layers::Image>&& aImage)
+  : webrtc::NativeHandleBuffer(aImage,
+                               aImage->GetSize().width,
+                               aImage->GetSize().height)
+  , mImage(Move(aImage))
+{
+}
+
+rtc::scoped_refptr<webrtc::VideoFrameBuffer>
+ImageBuffer::NativeToI420Buffer()
+{
+  RefPtr<layers::PlanarYCbCrImage> image = mImage->AsPlanarYCbCrImage();
+  if (!image) {
+    // TODO. YUV420 ReadBack, Image only provides a RGB readback.
+    return nullptr;
+  }
+  rtc::scoped_refptr<layers::PlanarYCbCrImage> refImage(image);
+  const layers::PlanarYCbCrData* data = image->GetData();
+  rtc::scoped_refptr<webrtc::VideoFrameBuffer> buf(
+    new rtc::RefCountedObject<webrtc::WrappedI420Buffer>(
+      data->mPicSize.width,
+      data->mPicSize.height,
+      data->mYChannel,
+      data->mYStride,
+      data->mCbChannel,
+      data->mCbCrStride,
+      data->mCrChannel,
+      data->mCbCrStride,
+      rtc::KeepRefUntilDone(refImage)));
+  return buf;
+}
+
+} // namespace mozilla
new file mode 100644
--- /dev/null
+++ b/media/webrtc/signaling/src/media-conduit/WebrtcMediaDataDecoderCodec.h
@@ -0,0 +1,83 @@
+/* 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/. */
+
+#ifndef WebrtcMediaDataDecoderCodec_h__
+#define WebrtcMediaDataDecoderCodec_h__
+
+#include "MediaConduitInterface.h"
+#include "MediaInfo.h"
+#include "MediaResult.h"
+#include "PlatformDecoderModule.h"
+#include "webrtc/common_video/include/video_frame_buffer.h"
+#include "webrtc/modules/video_coding/include/video_codec_interface.h"
+
+namespace webrtc {
+  class DecodedImageCallback;
+}
+namespace mozilla {
+namespace layers {
+  class Image;
+  class ImageContainer;
+}
+
+class PDMFactory;
+class TaskQueue;
+
+class ImageBuffer : public webrtc::NativeHandleBuffer
+{
+public:
+  explicit ImageBuffer(RefPtr<layers::Image>&& aImage);
+  rtc::scoped_refptr<VideoFrameBuffer> NativeToI420Buffer() override;
+
+private:
+  RefPtr<layers::Image> mImage;
+};
+
+class WebrtcMediaDataDecoder : public WebrtcVideoDecoder
+{
+public:
+  WebrtcMediaDataDecoder();
+
+  // Implement VideoDecoder interface.
+  uint64_t PluginID() const override { return 0; }
+
+  int32_t InitDecode(const webrtc::VideoCodec* codecSettings,
+                     int32_t numberOfCores) override;
+
+  int32_t Decode(const webrtc::EncodedImage& inputImage,
+                 bool missingFrames,
+                 const webrtc::RTPFragmentationHeader* fragmentation,
+                 const webrtc::CodecSpecificInfo* codecSpecificInfo = NULL,
+                 int64_t renderTimeMs = -1) override;
+
+  int32_t RegisterDecodeCompleteCallback(
+    webrtc::DecodedImageCallback* callback) override;
+
+  int32_t Release() override;
+
+private:
+  ~WebrtcMediaDataDecoder();
+  void QueueFrame(MediaRawData* aFrame);
+  AbstractThread* OwnerThread() const { return mTaskQueue; }
+  bool OnTaskQueue() const;
+
+  const RefPtr<TaskQueue> mTaskQueue;
+  const RefPtr<layers::ImageContainer> mImageContainer;
+  const RefPtr<PDMFactory> mFactory;
+  RefPtr<MediaDataDecoder> mDecoder;
+  webrtc::DecodedImageCallback* mCallback = nullptr;
+  VideoInfo mInfo;
+  TrackInfo::TrackType mTrackType;
+  bool mNeedKeyframe = true;
+  MozPromiseRequestHolder<MediaDataDecoder::DecodePromise> mDecodeRequest;
+
+  Monitor mMonitor;
+  // Members below are accessed via mMonitor
+  MediaResult mError = NS_OK;
+  MediaDataDecoder::DecodedData mResults;
+};
+
+} // namespace mozilla
+
+#endif // WebrtcMediaDataDecoderCodec_h__
deleted file mode 100644
--- a/media/webrtc/signaling/src/media-conduit/WebrtcOMXH264VideoCodec.cpp
+++ /dev/null
@@ -1,1253 +0,0 @@
-/* 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/. */
-
-#include "CSFLog.h"
-
-#include "WebrtcOMXH264VideoCodec.h"
-
-// Android/Stagefright
-#include <avc_utils.h>
-#include <binder/ProcessState.h>
-#include <foundation/ABuffer.h>
-#include <foundation/AMessage.h>
-#include <gui/Surface.h>
-#include <media/ICrypto.h>
-#include <media/stagefright/MediaCodec.h>
-#include <media/stagefright/MediaDefs.h>
-#include <media/stagefright/MediaErrors.h>
-#include <media/stagefright/MetaData.h>
-#include <OMX_Component.h>
-using namespace android;
-
-// WebRTC
-//#include "webrtc/common_video/interface/texture_video_frame.h"
-#include "webrtc/video_engine/include/vie_external_codec.h"
-#include "runnable_utils.h"
-
-// Gecko
-#if defined(MOZ_WIDGET_GONK) && ANDROID_VERSION >= 21
-#include "GonkBufferQueueProducer.h"
-#endif
-#include "GonkNativeWindow.h"
-#include "GrallocImages.h"
-#include "mozilla/Atomics.h"
-#include "mozilla/Mutex.h"
-#include "nsThreadUtils.h"
-#include "OMXCodecWrapper.h"
-#include "TextureClient.h"
-#include "mozilla/IntegerPrintfMacros.h"
-
-#define DEQUEUE_BUFFER_TIMEOUT_US (100 * 1000ll) // 100ms.
-#define START_DEQUEUE_BUFFER_TIMEOUT_US (10 * DEQUEUE_BUFFER_TIMEOUT_US) // 1s.
-#define DRAIN_THREAD_TIMEOUT_US  (1000 * 1000ll) // 1s.
-
-#define WOHVC_LOG_TAG "WebrtcOMXH264VideoCodec"
-#define CODEC_LOGV(...) CSFLogInfo(WOHVC_LOG_TAG, __VA_ARGS__)
-#define CODEC_LOGD(...) CSFLogDebug(WOHVC_LOG_TAG, __VA_ARGS__)
-#define CODEC_LOGI(...) CSFLogInfo(WOHVC_LOG_TAG, __VA_ARGS__)
-#define CODEC_LOGW(...) CSFLogWarn(WOHVC_LOG_TAG, __VA_ARGS__)
-#define CODEC_LOGE(...) CSFLogError(WOHVC_LOG_TAG, __VA_ARGS__)
-
-namespace mozilla {
-
-static const uint8_t kNALStartCode[] = { 0x00, 0x00, 0x00, 0x01 };
-enum {
-  kNALTypeIDR = 5,
-  kNALTypeSPS = 7,
-  kNALTypePPS = 8,
-};
-
-// NS_INLINE_DECL_THREADSAFE_REFCOUNTING() cannot be used directly in
-// ImageNativeHandle below because the return type of webrtc::NativeHandle
-// AddRef()/Release() conflicts with those defined in macro. To avoid another
-// copy/paste of ref-counting implementation here, this dummy base class
-// is created to proivde another level of indirection.
-class DummyRefCountBase {
-public:
-  // Use the name of real class for logging.
-  NS_INLINE_DECL_THREADSAFE_REFCOUNTING(DummyRefCountBase)
-protected:
-  // To make sure subclass will be deleted/destructed properly.
-  virtual ~DummyRefCountBase() {}
-};
-
-// This function implements 2 interafces:
-// 1. webrtc::NativeHandle: to wrap layers::Image object so decoded frames can
-//    be passed through WebRTC rendering pipeline using TextureVideoFrame.
-// 2. ImageHandle: for renderer to get the image object inside without knowledge
-//    about webrtc::NativeHandle.
-class ImageNativeHandle final
-  : public webrtc::NativeHandle
-  , public DummyRefCountBase
-{
-public:
-  ImageNativeHandle(layers::Image* aImage)
-    : mImage(aImage)
-  {}
-
-  // Implement webrtc::NativeHandle.
-  virtual void* GetHandle() override { return mImage.get(); }
-
-  virtual int AddRef() override
-  {
-    return DummyRefCountBase::AddRef();
-  }
-
-  virtual int Release() override
-  {
-    return DummyRefCountBase::Release();
-  }
-
-private:
-  RefPtr<layers::Image> mImage;
-};
-
-struct EncodedFrame
-{
-  uint32_t mWidth;
-  uint32_t mHeight;
-  uint32_t mTimestamp;
-  int64_t mRenderTimeMs;
-};
-
-static void
-ShutdownThread(nsCOMPtr<nsIThread>& aThread)
-{
-  aThread->Shutdown();
-}
-
-// Base runnable class to repeatly pull OMX output buffers in seperate thread.
-// How to use:
-// - implementing DrainOutput() to get output. Remember to return false to tell
-//   drain not to pop input queue.
-// - call QueueInput() to schedule a run to drain output. The input, aFrame,
-//   should contains corresponding info such as image size and timestamps for
-//   DrainOutput() implementation to construct data needed by encoded/decoded
-//   callbacks.
-// TODO: Bug 997110 - Revisit queue/drain logic. Current design assumes that
-//       encoder only generate one output buffer per input frame and won't work
-//       if encoder drops frames or generates multiple output per input.
-class OMXOutputDrain : public Runnable
-{
-public:
-  void Start() {
-    CODEC_LOGD("OMXOutputDrain starting");
-    MonitorAutoLock lock(mMonitor);
-    if (mThread == nullptr) {
-      NS_NewNamedThread("OMXOutputDrain", getter_AddRefs(mThread));
-    }
-    CODEC_LOGD("OMXOutputDrain started");
-    mEnding = false;
-    mThread->Dispatch(this, NS_DISPATCH_NORMAL);
-  }
-
-  void Stop() {
-    CODEC_LOGD("OMXOutputDrain stopping");
-    MonitorAutoLock lock(mMonitor);
-    mEnding = true;
-    lock.NotifyAll(); // In case Run() is waiting.
-
-    if (mThread != nullptr) {
-      MonitorAutoUnlock unlock(mMonitor);
-      CODEC_LOGD("OMXOutputDrain thread shutdown");
-      NS_DispatchToMainThread(
-        WrapRunnableNM<decltype(&ShutdownThread),
-                       nsCOMPtr<nsIThread> >(&ShutdownThread, mThread));
-      mThread = nullptr;
-    }
-    CODEC_LOGD("OMXOutputDrain stopped");
-  }
-
-  void QueueInput(const EncodedFrame& aFrame)
-  {
-    MonitorAutoLock lock(mMonitor);
-
-    MOZ_ASSERT(mThread);
-
-    mInputFrames.push(aFrame);
-    // Notify Run() about queued input and it can start working.
-    lock.NotifyAll();
-  }
-
-  NS_IMETHOD Run() override
-  {
-    MonitorAutoLock lock(mMonitor);
-    if (mEnding) {
-      return NS_OK;
-    }
-    MOZ_ASSERT(mThread);
-
-    while (true) {
-      if (mInputFrames.empty()) {
-        // Wait for new input.
-        lock.Wait();
-      }
-
-      if (mEnding) {
-        CODEC_LOGD("OMXOutputDrain Run() ending");
-        // Stop draining.
-        break;
-      }
-
-      MOZ_ASSERT(!mInputFrames.empty());
-      {
-        // Release monitor while draining because it's blocking.
-        MonitorAutoUnlock unlock(mMonitor);
-        DrainOutput();
-      }
-    }
-
-    CODEC_LOGD("OMXOutputDrain Ended");
-    return NS_OK;
-  }
-
-protected:
-  OMXOutputDrain()
-    : mMonitor("OMXOutputDrain monitor")
-    , mEnding(false)
-  {}
-
-  // Drain output buffer for input frame queue mInputFrames.
-  // mInputFrames contains info such as size and time of the input frames.
-  // We have to give a queue to handle encoder frame skips - we can input 10
-  // frames and get one back.  NOTE: any access of aInputFrames MUST be preceded
-  // locking mMonitor!
-
-  // Blocks waiting for decoded buffers, but for a limited period because
-  // we need to check for shutdown.
-  virtual bool DrainOutput() = 0;
-
-protected:
-  // This monitor protects all things below it, and is also used to
-  // wait/notify queued input.
-  Monitor mMonitor;
-  std::queue<EncodedFrame> mInputFrames;
-
-private:
-  // also protected by mMonitor
-  nsCOMPtr<nsIThread> mThread;
-  bool mEnding;
-};
-
-// Assumption: SPS is first paramset or is not present
-static bool IsParamSets(uint8_t* aData, size_t aSize)
-{
-  MOZ_ASSERT(aData && aSize > sizeof(kNALStartCode));
-  return (aData[sizeof(kNALStartCode)] & 0x1f) == kNALTypeSPS;
-}
-
-// get the length of any pre-pended SPS/PPS's
-static size_t ParamSetLength(uint8_t* aData, size_t aSize)
-{
-  const uint8_t* data = aData;
-  size_t size = aSize;
-  const uint8_t* nalStart = nullptr;
-  size_t nalSize = 0;
-  while (getNextNALUnit(&data, &size, &nalStart, &nalSize, true) == OK) {
-    if ((*nalStart & 0x1f) != kNALTypeSPS &&
-        (*nalStart & 0x1f) != kNALTypePPS) {
-      MOZ_ASSERT(nalStart - sizeof(kNALStartCode) >= aData);
-      return (nalStart - sizeof(kNALStartCode)) - aData; // SPS/PPS/iframe
-    }
-  }
-  return aSize; // it's only SPS/PPS
-}
-
-// H.264 decoder using stagefright.
-// It implements gonk native window callback to receive buffers from
-// MediaCodec::RenderOutputBufferAndRelease().
-class WebrtcOMXDecoder final : public GonkNativeWindowNewFrameCallback
-{
-  NS_INLINE_DECL_THREADSAFE_REFCOUNTING(WebrtcOMXDecoder)
-
-private:
-  virtual ~WebrtcOMXDecoder()
-  {
-    CODEC_LOGD("WebrtcOMXH264VideoDecoder:%p OMX destructor", this);
-    if (mStarted) {
-      Stop();
-    }
-    if (mCodec != nullptr) {
-      mCodec->release();
-      mCodec.clear();
-    }
-    mLooper.clear();
-  }
-
-public:
-  WebrtcOMXDecoder(const char* aMimeType,
-                   webrtc::DecodedImageCallback* aCallback)
-    : mWidth(0)
-    , mHeight(0)
-    , mStarted(false)
-    , mCallback(aCallback)
-    , mDecodedFrameLock("WebRTC decoded frame lock")
-    , mEnding(false)
-  {
-    // Create binder thread pool required by stagefright.
-    android::ProcessState::self()->startThreadPool();
-
-    mLooper = new ALooper;
-    mLooper->start();
-    CODEC_LOGD("WebrtcOMXH264VideoDecoder:%p creating decoder", this);
-    mCodec = MediaCodec::CreateByType(mLooper, aMimeType, false /* encoder */);
-    CODEC_LOGD("WebrtcOMXH264VideoDecoder:%p OMX created", this);
-  }
-
-  // Find SPS in input data and extract picture width and height if found.
-  static status_t ExtractPicDimensions(uint8_t* aData, size_t aSize,
-                                       int32_t* aWidth, int32_t* aHeight)
-  {
-    MOZ_ASSERT(aData && aSize > sizeof(kNALStartCode));
-    if ((aData[sizeof(kNALStartCode)] & 0x1f) != kNALTypeSPS) {
-      return ERROR_MALFORMED;
-    }
-    sp<ABuffer> sps = new ABuffer(&aData[sizeof(kNALStartCode)], aSize - sizeof(kNALStartCode));
-    FindAVCDimensions(sps, aWidth, aHeight);
-    return OK;
-  }
-
-  // Configure decoder using image width/height.
-  status_t ConfigureWithPicDimensions(int32_t aWidth, int32_t aHeight)
-  {
-    MOZ_ASSERT(mCodec != nullptr);
-    if (mCodec == nullptr) {
-      return INVALID_OPERATION;
-    }
-
-    CODEC_LOGD("OMX:%p decoder width:%d height:%d", this, aWidth, aHeight);
-
-    sp<AMessage> config = new AMessage();
-    config->setString("mime", MEDIA_MIMETYPE_VIDEO_AVC);
-    config->setInt32("width", aWidth);
-    config->setInt32("height", aHeight);
-    mWidth = aWidth;
-    mHeight = aHeight;
-
-    sp<Surface> surface = nullptr;
-#if defined(MOZ_WIDGET_GONK) && ANDROID_VERSION >= 21
-    sp<IGraphicBufferProducer> producer;
-    sp<IGonkGraphicBufferConsumer> consumer;
-    GonkBufferQueue::createBufferQueue(&producer, &consumer);
-    mNativeWindow = new GonkNativeWindow(consumer);
-#else
-    mNativeWindow = new GonkNativeWindow();
-#endif
-    if (mNativeWindow.get()) {
-      // listen to buffers queued by MediaCodec::RenderOutputBufferAndRelease().
-      mNativeWindow->setNewFrameCallback(this);
-      // XXX remove buffer changes after a better solution lands - bug 1009420
-#if defined(MOZ_WIDGET_GONK) && ANDROID_VERSION >= 21
-      static_cast<GonkBufferQueueProducer*>(producer.get())->setSynchronousMode(false);
-      // More spare buffers to avoid OMX decoder waiting for native window
-      consumer->setMaxAcquiredBufferCount(WEBRTC_OMX_H264_MIN_DECODE_BUFFERS);
-      surface = new Surface(producer);
-#else
-      sp<GonkBufferQueue> bq = mNativeWindow->getBufferQueue();
-      bq->setSynchronousMode(false);
-      // More spare buffers to avoid OMX decoder waiting for native window
-      bq->setMaxAcquiredBufferCount(WEBRTC_OMX_H264_MIN_DECODE_BUFFERS);
-      surface = new Surface(bq);
-#endif
-    }
-    status_t result = mCodec->configure(config, surface, nullptr, 0);
-    if (result == OK) {
-      CODEC_LOGD("OMX:%p decoder configured", this);
-      result = Start();
-    }
-    return result;
-  }
-
-  status_t
-  FillInput(const webrtc::EncodedImage& aEncoded, bool aIsFirstFrame,
-            int64_t& aRenderTimeMs)
-  {
-    MOZ_ASSERT(mCodec != nullptr && aEncoded._buffer && aEncoded._length > 0);
-    if (mCodec == nullptr || !aEncoded._buffer || aEncoded._length == 0) {
-      return INVALID_OPERATION;
-    }
-
-    // Break input encoded data into NALUs and send each one to decode.
-    // 8x10 decoder doesn't allow picture coding NALs to be in the same buffer
-    // with SPS/PPS (BUFFER_FLAG_CODECCONFIG) per QC
-    const uint8_t* data = aEncoded._buffer;
-    size_t size = aEncoded._length;
-    const uint8_t* nalStart = nullptr;
-    size_t nalSize = 0;
-    status_t err = OK;
-
-    // this returns a pointer to the NAL byte (after the StartCode)
-    while (getNextNALUnit(&data, &size, &nalStart, &nalSize, true) == OK) {
-      // Individual NALU inherits metadata from input encoded data.
-      webrtc::EncodedImage nalu(aEncoded);
-
-      nalu._buffer = const_cast<uint8_t*>(nalStart) - sizeof(kNALStartCode);
-      MOZ_ASSERT(nalu._buffer >= aEncoded._buffer);
-      nalu._length = nalSize + sizeof(kNALStartCode);
-      MOZ_ASSERT(nalu._buffer + nalu._length <= aEncoded._buffer + aEncoded._length);
-
-      size_t index;
-      err = mCodec->dequeueInputBuffer(&index,
-                                       aIsFirstFrame ? START_DEQUEUE_BUFFER_TIMEOUT_US : DEQUEUE_BUFFER_TIMEOUT_US);
-      if (err != OK) {
-        if (err != -EAGAIN) {
-          CODEC_LOGE("decode dequeue input buffer error:%d", err);
-        } else {
-          CODEC_LOGE("decode dequeue 100ms without a buffer (EAGAIN)");
-        }
-        return err;
-      }
-
-      // Prepend start code to buffer.
-      MOZ_ASSERT(memcmp(nalu._buffer, kNALStartCode, sizeof(kNALStartCode)) == 0);
-      const sp<ABuffer>& omxIn = mInputBuffers.itemAt(index);
-      MOZ_ASSERT(omxIn->capacity() >= nalu._length);
-      omxIn->setRange(0, nalu._length);
-      // Copying is needed because MediaCodec API doesn't support externally
-      // allocated buffer as input.
-      uint8_t* dst = omxIn->data();
-      memcpy(dst, nalu._buffer, nalu._length);
-      int64_t inputTimeUs = (nalu._timeStamp * 1000ll) / 90; // 90kHz -> us.
-      // Assign input flags according to input buffer NALU and frame types.
-      uint32_t flags;
-      int nalType = dst[sizeof(kNALStartCode)] & 0x1f;
-      switch (nalType) {
-        case kNALTypeSPS:
-        case kNALTypePPS:
-          flags = MediaCodec::BUFFER_FLAG_CODECCONFIG;
-          break;
-        case kNALTypeIDR:
-          flags = MediaCodec::BUFFER_FLAG_SYNCFRAME;
-          break;
-        default:
-          flags = 0;
-          break;
-      }
-      CODEC_LOGD("Decoder input: %d bytes (NAL 0x%02x), time %lld (%u), flags 0x%x",
-                 nalu._length, dst[sizeof(kNALStartCode)], inputTimeUs, nalu._timeStamp, flags);
-      err = mCodec->queueInputBuffer(index, 0, nalu._length, inputTimeUs, flags);
-      if (err == OK && !(flags & MediaCodec::BUFFER_FLAG_CODECCONFIG)) {
-        if (mOutputDrain == nullptr) {
-          mOutputDrain = new OutputDrain(this);
-          mOutputDrain->Start();
-        }
-        EncodedFrame frame;
-        frame.mWidth = mWidth;
-        frame.mHeight = mHeight;
-        frame.mTimestamp = nalu._timeStamp;
-        frame.mRenderTimeMs = aRenderTimeMs;
-        mOutputDrain->QueueInput(frame);
-      }
-    }
-
-    return err;
-  }
-
-  status_t
-  DrainOutput(std::queue<EncodedFrame>& aInputFrames, Monitor& aMonitor)
-  {
-    MOZ_ASSERT(mCodec != nullptr);
-    if (mCodec == nullptr) {
-      return INVALID_OPERATION;
-    }
-
-    size_t index = 0;
-    size_t outOffset = 0;
-    size_t outSize = 0;
-    int64_t outTime = -1ll;
-    uint32_t outFlags = 0;
-    status_t err = mCodec->dequeueOutputBuffer(&index, &outOffset, &outSize,
-                                               &outTime, &outFlags,
-                                               DRAIN_THREAD_TIMEOUT_US);
-    switch (err) {
-      case OK:
-        break;
-      case -EAGAIN:
-        // Not an error: output not available yet. Try later.
-        CODEC_LOGI("decode dequeue OMX output buffer timed out. Try later.");
-        return err;
-      case INFO_FORMAT_CHANGED:
-        // Not an error: will get this value when OMX output buffer is enabled,
-        // or when input size changed.
-        CODEC_LOGD("decode dequeue OMX output buffer format change");
-        return err;
-      case INFO_OUTPUT_BUFFERS_CHANGED:
-        // Not an error: will get this value when OMX output buffer changed
-        // (probably because of input size change).
-        CODEC_LOGD("decode dequeue OMX output buffer change");
-        err = mCodec->getOutputBuffers(&mOutputBuffers);
-        MOZ_ASSERT(err == OK);
-        return INFO_OUTPUT_BUFFERS_CHANGED;
-      default:
-        CODEC_LOGE("decode dequeue OMX output buffer error:%d", err);
-        // Return OK to instruct OutputDrain to drop input from queue.
-        MonitorAutoLock lock(aMonitor);
-        aInputFrames.pop();
-        return OK;
-    }
-
-    CODEC_LOGD("Decoder output: %d bytes, offset %u, time %lld, flags 0x%x",
-               outSize, outOffset, outTime, outFlags);
-    if (mCallback) {
-      EncodedFrame frame;
-      {
-        MonitorAutoLock lock(aMonitor);
-        frame = aInputFrames.front();
-        aInputFrames.pop();
-      }
-      {
-        // Store info of this frame. OnNewFrame() will need the timestamp later.
-        MutexAutoLock lock(mDecodedFrameLock);
-        if (mEnding) {
-          mCodec->releaseOutputBuffer(index);
-          return err;
-        }
-        mDecodedFrames.push(frame);
-      }
-      // Ask codec to queue buffer back to native window. OnNewFrame() will be
-      // called.
-      mCodec->renderOutputBufferAndRelease(index);
-      // Once consumed, buffer will be queued back to GonkNativeWindow for codec
-      // to dequeue/use.
-    } else {
-      mCodec->releaseOutputBuffer(index);
-    }
-
-    return err;
-  }
-
-  // Will be called when MediaCodec::RenderOutputBufferAndRelease() returns
-  // buffers back to native window for rendering.
-  void OnNewFrame() override
-  {
-    RefPtr<layers::TextureClient> buffer = mNativeWindow->getCurrentBuffer();
-    if (!buffer) {
-      CODEC_LOGE("Decoder NewFrame: Get null buffer");
-      return;
-    }
-
-    gfx::IntSize picSize(buffer->GetSize());
-    nsAutoPtr<layers::GrallocImage> grallocImage(new layers::GrallocImage());
-    grallocImage->AdoptData(buffer, picSize);
-
-    // Get timestamp of the frame about to render.
-    int64_t timestamp = -1;
-    int64_t renderTimeMs = -1;
-    {
-      MutexAutoLock lock(mDecodedFrameLock);
-      if (mDecodedFrames.empty()) {
-        return;
-      }
-      EncodedFrame decoded = mDecodedFrames.front();
-      timestamp = decoded.mTimestamp;
-      renderTimeMs = decoded.mRenderTimeMs;
-      mDecodedFrames.pop();
-    }
-    MOZ_ASSERT(timestamp >= 0 && renderTimeMs >= 0);
-
-    CODEC_LOGD("Decoder NewFrame: %dx%d, timestamp %lld, renderTimeMs %lld",
-               picSize.width, picSize.height, timestamp, renderTimeMs);
-
-    nsAutoPtr<webrtc::VideoFrame> videoFrame(new webrtc::VideoFrame(
-      new ImageNativeHandle(grallocImage.forget()),
-      picSize.width,
-      picSize.height,
-      timestamp,
-      renderTimeMs));
-    if (videoFrame != nullptr) {
-      mCallback->Decoded(*videoFrame);
-    }
-  }
-
-private:
-  class OutputDrain : public OMXOutputDrain
-  {
-  public:
-    OutputDrain(WebrtcOMXDecoder* aOMX)
-      : OMXOutputDrain()
-      , mOMX(aOMX)
-    {}
-
-  protected:
-    virtual bool DrainOutput() override
-    {
-      return (mOMX->DrainOutput(mInputFrames, mMonitor) == OK);
-    }
-
-  private:
-    WebrtcOMXDecoder* mOMX;
-  };
-
-  status_t Start()
-  {
-    MOZ_ASSERT(!mStarted);
-    if (mStarted) {
-      return OK;
-    }
-
-    {
-      MutexAutoLock lock(mDecodedFrameLock);
-      mEnding = false;
-    }
-    status_t err = mCodec->start();
-    if (err == OK) {
-      mStarted = true;
-      mCodec->getInputBuffers(&mInputBuffers);
-      mCodec->getOutputBuffers(&mOutputBuffers);
-    }
-
-    return err;
-  }
-
-  status_t Stop()
-  {
-    MOZ_ASSERT(mStarted);
-    if (!mStarted) {
-      return OK;
-    }
-
-    CODEC_LOGD("OMXOutputDrain decoder stopping");
-    // Drop all 'pending to render' frames.
-    {
-      MutexAutoLock lock(mDecodedFrameLock);
-      mEnding = true;
-      while (!mDecodedFrames.empty()) {
-        mDecodedFrames.pop();
-      }
-    }
-
-    if (mOutputDrain != nullptr) {
-      CODEC_LOGD("decoder's OutputDrain stopping");
-      mOutputDrain->Stop();
-      mOutputDrain = nullptr;
-    }
-
-    status_t err = mCodec->stop();
-    if (err == OK) {
-      mInputBuffers.clear();
-      mOutputBuffers.clear();
-      mStarted = false;
-    } else {
-      MOZ_ASSERT(false);
-    }
-    CODEC_LOGD("OMXOutputDrain decoder stopped");
-    return err;
-  }
-
-  sp<ALooper> mLooper;
-  sp<MediaCodec> mCodec; // OMXCodec
-  int mWidth;
-  int mHeight;
-  android::Vector<sp<ABuffer> > mInputBuffers;
-  android::Vector<sp<ABuffer> > mOutputBuffers;
-  bool mStarted;
-
-  sp<GonkNativeWindow> mNativeWindow;
-
-  RefPtr<OutputDrain> mOutputDrain;
-  webrtc::DecodedImageCallback* mCallback;
-
-  Mutex mDecodedFrameLock; // To protect mDecodedFrames and mEnding
-  std::queue<EncodedFrame> mDecodedFrames;
-  bool mEnding;
-};
-
-class EncOutputDrain : public OMXOutputDrain
-{
-public:
-  EncOutputDrain(OMXVideoEncoder* aOMX, webrtc::EncodedImageCallback* aCallback)
-    : OMXOutputDrain()
-    , mOMX(aOMX)
-    , mCallback(aCallback)
-    , mIsPrevFrameParamSets(false)
-  {}
-
-protected:
-  virtual bool DrainOutput() override
-  {
-    nsTArray<uint8_t> output;
-    int64_t timeUs = -1ll;
-    int flags = 0;
-    nsresult rv = mOMX->GetNextEncodedFrame(&output, &timeUs, &flags,
-                                            DRAIN_THREAD_TIMEOUT_US);
-    if (NS_WARN_IF(NS_FAILED(rv))) {
-      // Fail to get encoded frame. The corresponding input frame should be
-      // removed.
-      // We'll treat this like a skipped frame
-      return true;
-    }
-
-    if (output.Length() == 0) {
-      // No encoded data yet. Try later.
-      CODEC_LOGD("OMX: (encode no output available this time)");
-      return false;
-    }
-
-    // Conversion to us rounds down, so we need to round up for us->90KHz
-    uint32_t target_timestamp = (timeUs * 90ll + 999) / 1000; // us -> 90KHz
-    // 8x10 v2.0 encoder doesn't set this reliably:
-    //bool isParamSets = (flags & MediaCodec::BUFFER_FLAG_CODECCONFIG);
-    // Assume that SPS/PPS will be at the start of any buffer
-    // Assume PPS will not be in a separate buffer - SPS/PPS or SPS/PPS/iframe
-    bool isParamSets = IsParamSets(output.Elements(), output.Length());
-    bool isIFrame = (flags & MediaCodec::BUFFER_FLAG_SYNCFRAME);
-    CODEC_LOGD("OMX: encoded frame (%d): time %lld (%u), flags x%x",
-               output.Length(), timeUs, target_timestamp, flags);
-    // Should not be parameter sets and I-frame at the same time.
-    // Except that it is possible, apparently, after an encoder re-config (bug 1063883)
-    // MOZ_ASSERT(!(isParamSets && isIFrame));
-
-    if (mCallback) {
-      // Implementation here assumes encoder output to be a buffer containing
-      // parameter sets(SPS + PPS) followed by a series of buffers, each for
-      // one input frame.
-      // TODO: handle output violating this assumpton in bug 997110.
-      webrtc::EncodedImage encoded(output.Elements(), output.Length(),
-                                   output.Capacity());
-      encoded._frameType = (isParamSets || isIFrame) ?
-                           webrtc::kKeyFrame : webrtc::kDeltaFrame;
-      EncodedFrame input_frame;
-      {
-        MonitorAutoLock lock(mMonitor);
-        // will sps/pps have the same timestamp as their iframe? Initial one on 8x10 has
-        // 0 timestamp.
-        if (isParamSets) {
-          // Let's assume it was the first item in the queue, but leave it there since an
-          // IDR will follow
-          input_frame = mInputFrames.front();
-        } else {
-          do {
-            if (mInputFrames.empty()) {
-              // Let's assume it was the last item in the queue, but leave it there
-              mInputFrames.push(input_frame);
-              CODEC_LOGE("OMX: encoded timestamp %u which doesn't match input queue!! (head %u)",
-                         target_timestamp, input_frame.mTimestamp);
-              break;
-            }
-
-            input_frame = mInputFrames.front();
-            mInputFrames.pop();
-            if (input_frame.mTimestamp != target_timestamp) {
-              CODEC_LOGD("OMX: encoder skipped frame timestamp %u", input_frame.mTimestamp);
-            }
-          } while (input_frame.mTimestamp != target_timestamp);
-        }
-      }
-
-      encoded._encodedWidth = input_frame.mWidth;
-      encoded._encodedHeight = input_frame.mHeight;
-      encoded._timeStamp = input_frame.mTimestamp;
-      encoded.capture_time_ms_ = input_frame.mRenderTimeMs;
-      encoded._completeFrame = true;
-
-      CODEC_LOGD("Encoded frame: %d bytes, %dx%d, is_param %d, is_iframe %d, timestamp %u, captureTimeMs %" PRIu64,
-                 encoded._length, encoded._encodedWidth, encoded._encodedHeight,
-                 isParamSets, isIFrame, encoded._timeStamp, encoded.capture_time_ms_);
-      // Prepend SPS/PPS to I-frames unless they were sent last time.
-      SendEncodedDataToCallback(encoded, isIFrame && !mIsPrevFrameParamSets && !isParamSets);
-      // This will be true only for the frame following a paramset block!  So if we're
-      // working with a correct encoder that generates SPS/PPS then iframe always, we
-      // won't try to insert.  (also, don't set if we get SPS/PPS/iframe in one buffer)
-      mIsPrevFrameParamSets = isParamSets && !isIFrame;
-      if (isParamSets) {
-        // copy off the param sets for inserting later
-        mParamSets.Clear();
-        // since we may have SPS/PPS or SPS/PPS/iframe
-        size_t length = ParamSetLength(encoded._buffer, encoded._length);
-        MOZ_ASSERT(length > 0);
-        mParamSets.AppendElements(encoded._buffer, length);
-      }
-    }
-
-    return !isParamSets; // not really needed anymore
-  }
-
-private:
-  // Send encoded data to callback.The data will be broken into individual NALUs
-  // if necessary and sent to callback one by one. This function can also insert
-  // SPS/PPS NALUs in front of input data if requested.
-  void SendEncodedDataToCallback(webrtc::EncodedImage& aEncodedImage,
-                                 bool aPrependParamSets)
-  {
-    if (aPrependParamSets) {
-      webrtc::EncodedImage prepend(aEncodedImage);
-      // Insert current parameter sets in front of the input encoded data.
-      MOZ_ASSERT(mParamSets.Length() > sizeof(kNALStartCode)); // Start code + ...
-      prepend._length = mParamSets.Length();
-      prepend._buffer = mParamSets.Elements();
-      // Break into NALUs and send.
-      CODEC_LOGD("Prepending SPS/PPS: %d bytes, timestamp %u, captureTimeMs %" PRIu64,
-                 prepend._length, prepend._timeStamp, prepend.capture_time_ms_);
-      SendEncodedDataToCallback(prepend, false);
-    }
-
-    struct nal_entry {
-      uint32_t offset;
-      uint32_t size;
-    };
-    AutoTArray<nal_entry, 1> nals;
-
-    // Break input encoded data into NALUs and send each one to callback.
-    const uint8_t* data = aEncodedImage._buffer;
-    size_t size = aEncodedImage._length;
-    const uint8_t* nalStart = nullptr;
-    size_t nalSize = 0;
-    while (getNextNALUnit(&data, &size, &nalStart, &nalSize, true) == OK) {
-      // XXX optimize by making buffer an offset
-      nal_entry nal = {((uint32_t) (nalStart - aEncodedImage._buffer)), (uint32_t) nalSize};
-      nals.AppendElement(nal);
-    }
-
-    size_t num_nals = nals.Length();
-    if (num_nals > 0) {
-      webrtc::RTPFragmentationHeader fragmentation;
-      fragmentation.VerifyAndAllocateFragmentationHeader(num_nals);
-      for (size_t i = 0; i < num_nals; i++) {
-        fragmentation.fragmentationOffset[i] = nals[i].offset;
-        fragmentation.fragmentationLength[i] = nals[i].size;
-      }
-      webrtc::EncodedImage unit(aEncodedImage);
-      unit._completeFrame = true;
-
-      mCallback->Encoded(unit, nullptr, &fragmentation);
-    }
-  }
-
-  OMXVideoEncoder* mOMX;
-  webrtc::EncodedImageCallback* mCallback;
-  bool mIsPrevFrameParamSets;
-  nsTArray<uint8_t> mParamSets;
-};
-
-// Encoder.
-WebrtcOMXH264VideoEncoder::WebrtcOMXH264VideoEncoder()
-  : mOMX(nullptr)
-  , mCallback(nullptr)
-  , mWidth(0)
-  , mHeight(0)
-  , mFrameRate(0)
-  , mBitRateKbps(0)
-#ifdef OMX_IDR_NEEDED_FOR_BITRATE
-  , mBitRateAtLastIDR(0)
-#endif
-  , mOMXConfigured(false)
-  , mOMXReconfigure(false)
-{
-  mReservation = new OMXCodecReservation(true);
-  CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p constructed", this);
-}
-
-int32_t
-WebrtcOMXH264VideoEncoder::InitEncode(const webrtc::VideoCodec* aCodecSettings,
-                                      int32_t aNumOfCores,
-                                      size_t aMaxPayloadSize)
-{
-  CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p init", this);
-
-  if (mOMX == nullptr) {
-    nsAutoPtr<OMXVideoEncoder> omx(OMXCodecWrapper::CreateAVCEncoder());
-    if (NS_WARN_IF(omx == nullptr)) {
-      return WEBRTC_VIDEO_CODEC_ERROR;
-    }
-    mOMX = omx.forget();
-    CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p OMX created", this);
-  }
-
-  if (!mReservation->ReserveOMXCodec()) {
-    CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p Encoder in use", this);
-    mOMX = nullptr;
-    return WEBRTC_VIDEO_CODEC_ERROR;
-  }
-
-  // Defer configuration until 1st frame is received because this function will
-  // be called more than once, and unfortunately with incorrect setting values
-  // at first.
-  mWidth = aCodecSettings->width;
-  mHeight = aCodecSettings->height;
-  mFrameRate = aCodecSettings->maxFramerate;
-  mBitRateKbps = aCodecSettings->startBitrate;
-  // XXX handle maxpayloadsize (aka mode 0/1)
-
-  CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p OMX Encoder reserved", this);
-  return WEBRTC_VIDEO_CODEC_OK;
-}
-
-int32_t
-WebrtcOMXH264VideoEncoder::Encode(const webrtc::VideoFrame& aInputImage,
-                                  const webrtc::CodecSpecificInfo* aCodecSpecificInfo,
-                                  const std::vector<webrtc::FrameType>* aFrameTypes)
-{
-  MOZ_ASSERT(mOMX != nullptr);
-  if (mOMX == nullptr) {
-    return WEBRTC_VIDEO_CODEC_ERROR;
-  }
-
-  // Have to reconfigure for resolution or framerate changes :-(
-  // ~220ms initial configure on 8x10, 50-100ms for re-configure it appears
-  // XXX drop frames while this is happening?
-  if (aInputImage.width() < 0 || (uint32_t)aInputImage.width() != mWidth ||
-      aInputImage.height() < 0 || (uint32_t)aInputImage.height() != mHeight) {
-    mWidth = aInputImage.width();
-    mHeight = aInputImage.height();
-    mOMXReconfigure = true;
-  }
-
-  if (!mOMXConfigured || mOMXReconfigure) {
-    if (mOMXConfigured) {
-      CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p reconfiguring encoder %dx%d @ %u fps",
-                 this, mWidth, mHeight, mFrameRate);
-      mOMXConfigured = false;
-    }
-    mOMXReconfigure = false;
-    // XXX This can take time.  Encode() likely assumes encodes are queued "quickly" and
-    // don't block the input too long.  Frames may build up.
-
-    // XXX take from negotiated SDP in codecSpecific data
-    OMX_VIDEO_AVCLEVELTYPE level = OMX_VIDEO_AVCLevel3;
-    // OMX_Video_ControlRateConstant is not supported on QC 8x10
-    OMX_VIDEO_CONTROLRATETYPE bitrateMode = OMX_Video_ControlRateConstantSkipFrames;
-
-    // Set up configuration parameters for AVC/H.264 encoder.
-    sp<AMessage> format = new AMessage;
-    // Fixed values
-    format->setString("mime", MEDIA_MIMETYPE_VIDEO_AVC);
-    // XXX We should only set to < infinity if we're not using any recovery RTCP options
-    // However, we MUST set it to a lower value because the 8x10 rate controller
-    // only changes rate at GOP boundaries.... but it also changes rate on requested GOPs
-
-    // Too long and we have very low bitrates for the first second or two... plus
-    // bug 1014921 means we have to force them every ~3 seconds or less.
-    format->setInt32("i-frame-interval", 4 /* seconds */);
-    // See mozilla::layers::GrallocImage, supports YUV 4:2:0, CbCr width and
-    // height is half that of Y
-    format->setInt32("color-format", OMX_COLOR_FormatYUV420SemiPlanar);
-    format->setInt32("profile", OMX_VIDEO_AVCProfileBaseline);
-    format->setInt32("level", level);
-    format->setInt32("bitrate-mode", bitrateMode);
-    format->setInt32("store-metadata-in-buffers", 0);
-    // XXX Unfortunately, 8x10 doesn't support this, but ask anyways
-    format->setInt32("prepend-sps-pps-to-idr-frames", 1);
-    // Input values.
-    format->setInt32("width", mWidth);
-    format->setInt32("height", mHeight);
-    format->setInt32("stride", mWidth);
-    format->setInt32("slice-height", mHeight);
-    format->setInt32("frame-rate", mFrameRate);
-    format->setInt32("bitrate", mBitRateKbps*1000);
-
-    CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p configuring encoder %dx%d @ %d fps, rate %d kbps",
-               this, mWidth, mHeight, mFrameRate, mBitRateKbps);
-    nsresult rv = mOMX->ConfigureDirect(format,
-                                        OMXVideoEncoder::BlobFormat::AVC_NAL);
-    if (NS_WARN_IF(NS_FAILED(rv))) {
-      CODEC_LOGE("WebrtcOMXH264VideoEncoder:%p FAILED configuring encoder %d", this, int(rv));
-      return WEBRTC_VIDEO_CODEC_ERROR;
-    }
-    mOMXConfigured = true;
-#ifdef OMX_IDR_NEEDED_FOR_BITRATE
-    mLastIDRTime = TimeStamp::Now();
-    mBitRateAtLastIDR = mBitRateKbps;
-#endif
-  }
-
-  if (aFrameTypes && aFrameTypes->size() &&
-      ((*aFrameTypes)[0] == webrtc::kKeyFrame)) {
-    mOMX->RequestIDRFrame();
-#ifdef OMX_IDR_NEEDED_FOR_BITRATE
-    mLastIDRTime = TimeStamp::Now();
-    mBitRateAtLastIDR = mBitRateKbps;
-  } else if (mBitRateKbps != mBitRateAtLastIDR) {
-    // 8x10 OMX codec requires a keyframe to shift bitrates!
-    TimeStamp now = TimeStamp::Now();
-    if (mLastIDRTime.IsNull()) {
-      // paranoia
-      mLastIDRTime = now;
-    }
-    int32_t timeSinceLastIDR = (now - mLastIDRTime).ToMilliseconds();
-
-    // Balance asking for IDRs too often against direction and amount of bitrate change.
-
-    // HACK for bug 1014921: 8x10 has encode/decode mismatches that build up errors
-    // if you go too long without an IDR.  In normal use, bitrate will change often
-    // enough to never hit this time limit.
-    if ((timeSinceLastIDR > 3000) ||
-        (mBitRateKbps < (mBitRateAtLastIDR * 8)/10) ||
-        (timeSinceLastIDR < 300 && mBitRateKbps < (mBitRateAtLastIDR * 9)/10) ||
-        (timeSinceLastIDR < 1000 && mBitRateKbps < (mBitRateAtLastIDR * 97)/100) ||
-        (timeSinceLastIDR >= 1000 && mBitRateKbps < mBitRateAtLastIDR) ||
-        (mBitRateKbps > (mBitRateAtLastIDR * 15)/10) ||
-        (timeSinceLastIDR < 500 && mBitRateKbps > (mBitRateAtLastIDR * 13)/10) ||
-        (timeSinceLastIDR < 1000 && mBitRateKbps > (mBitRateAtLastIDR * 11)/10) ||
-        (timeSinceLastIDR >= 1000 && mBitRateKbps > mBitRateAtLastIDR)) {
-      CODEC_LOGD("Requesting IDR for bitrate change from %u to %u (time since last idr %dms)",
-                 mBitRateAtLastIDR, mBitRateKbps, timeSinceLastIDR);
-
-      mOMX->RequestIDRFrame();
-      mLastIDRTime = now;
-      mBitRateAtLastIDR = mBitRateKbps;
-    }
-#endif
-  }
-
-  // Wrap VideoFrame input with PlanarYCbCrImage for OMXVideoEncoder.
-  layers::PlanarYCbCrData yuvData;
-  yuvData.mYChannel = const_cast<uint8_t*>(aInputImage.buffer(webrtc::kYPlane));
-  yuvData.mYSize = gfx::IntSize(aInputImage.width(), aInputImage.height());
-  yuvData.mYStride = aInputImage.stride(webrtc::kYPlane);
-  MOZ_ASSERT(aInputImage.stride(webrtc::kUPlane) == aInputImage.stride(webrtc::kVPlane));
-  yuvData.mCbCrStride = aInputImage.stride(webrtc::kUPlane);
-  yuvData.mCbChannel = const_cast<uint8_t*>(aInputImage.buffer(webrtc::kUPlane));
-  yuvData.mCrChannel = const_cast<uint8_t*>(aInputImage.buffer(webrtc::kVPlane));
-  yuvData.mCbCrSize = gfx::IntSize((yuvData.mYSize.width + 1) / 2,
-                                   (yuvData.mYSize.height + 1) / 2);
-  yuvData.mPicSize = yuvData.mYSize;
-  yuvData.mStereoMode = StereoMode::MONO;
-  layers::RecyclingPlanarYCbCrImage img(nullptr);
-  // AdoptData() doesn't need AllocateAndGetNewBuffer(); OMXVideoEncoder is ok with this
-  img.AdoptData(yuvData);
-
-  CODEC_LOGD("Encode frame: %dx%d, timestamp %u (%lld), renderTimeMs %" PRIu64,
-             aInputImage.width(), aInputImage.height(),
-             aInputImage.timestamp(), aInputImage.timestamp() * 1000ll / 90,
-             aInputImage.render_time_ms());
-
-  nsresult rv = mOMX->Encode(&img,
-                             yuvData.mYSize.width,
-                             yuvData.mYSize.height,
-                             aInputImage.timestamp() * 1000ll / 90, // 90kHz -> us.
-                             0);
-  if (rv == NS_OK) {
-    if (mOutputDrain == nullptr) {
-      mOutputDrain = new EncOutputDrain(mOMX, mCallback);
-      mOutputDrain->Start();
-    }
-    EncodedFrame frame;
-    frame.mWidth = mWidth;
-    frame.mHeight = mHeight;
-    frame.mTimestamp = aInputImage.timestamp();
-    frame.mRenderTimeMs = aInputImage.render_time_ms();
-    mOutputDrain->QueueInput(frame);
-  }
-
-  return (rv == NS_OK) ? WEBRTC_VIDEO_CODEC_OK : WEBRTC_VIDEO_CODEC_ERROR;
-}
-
-int32_t
-WebrtcOMXH264VideoEncoder::RegisterEncodeCompleteCallback(
-    webrtc::EncodedImageCallback* aCallback)
-{
-  CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p set callback:%p", this, aCallback);
-  MOZ_ASSERT(aCallback);
-  mCallback = aCallback;
-
-  return WEBRTC_VIDEO_CODEC_OK;
-}
-
-int32_t
-WebrtcOMXH264VideoEncoder::Release()
-{
-  CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p will be released", this);
-
-  if (mOutputDrain != nullptr) {
-    mOutputDrain->Stop();
-    mOutputDrain = nullptr;
-  }
-  mOMXConfigured = false;
-  bool hadOMX = !!mOMX;
-  mOMX = nullptr;
-  if (hadOMX) {
-    mReservation->ReleaseOMXCodec();
-  }
-  CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p released", this);
-
-  return WEBRTC_VIDEO_CODEC_OK;
-}
-
-WebrtcOMXH264VideoEncoder::~WebrtcOMXH264VideoEncoder()
-{
-  CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p will be destructed", this);
-
-  Release();
-}
-
-// Inform the encoder of the new packet loss rate and the round-trip time of
-// the network. aPacketLossRate is fraction lost and can be 0~255
-// (255 means 100% lost).
-// Note: stagefright doesn't handle these parameters.
-int32_t
-WebrtcOMXH264VideoEncoder::SetChannelParameters(uint32_t aPacketLossRate,
-                                                int64_t aRoundTripTimeMs)
-{
-  CODEC_LOGD("WebrtcOMXH264VideoEncoder:%p set channel packet loss:%u, rtt:%" PRIi64,
-             this, aPacketLossRate, aRoundTripTimeMs);
-
-  return WEBRTC_VIDEO_CODEC_OK;
-}
-
-// TODO: Bug 997567. Find the way to support frame rate change.
-int32_t
-WebrtcOMXH264VideoEncoder::SetRates(uint32_t aBitRateKbps, uint32_t aFrameRate)
-{
-  CODEC_LOGE("WebrtcOMXH264VideoEncoder:%p set bitrate:%u, frame rate:%u (%u))",
-             this, aBitRateKbps, aFrameRate, mFrameRate);
-  MOZ_ASSERT(mOMX != nullptr);
-  if (mOMX == nullptr) {
-    return WEBRTC_VIDEO_CODEC_UNINITIALIZED;
-  }
-
-  // XXX Should use StageFright framerate change, perhaps only on major changes of framerate.
-
-  // Without Stagefright support, Algorithm should be:
-  // if (frameRate < 50% of configured) {
-  //   drop framerate to next step down that includes current framerate within 50%
-  // } else if (frameRate > configured) {
-  //   change config to next step up that includes current framerate
-  // }
-#if !defined(TEST_OMX_FRAMERATE_CHANGES)
-  if (aFrameRate > mFrameRate ||
-      aFrameRate < mFrameRate/2) {
-    uint32_t old_rate = mFrameRate;
-    if (aFrameRate >= 15) {
-      mFrameRate = 30;
-    } else if (aFrameRate >= 10) {
-      mFrameRate = 20;
-    } else if (aFrameRate >= 8) {
-      mFrameRate = 15;
-    } else /* if (aFrameRate >= 5)*/ {
-      // don't go lower; encoder may not be stable
-      mFrameRate = 10;
-    }
-    if (mFrameRate < aFrameRate) { // safety
-      mFrameRate = aFrameRate;
-    }
-    if (old_rate != mFrameRate) {
-      mOMXReconfigure = true;  // force re-configure on next frame
-    }
-  }
-#else
-  // XXX for testing, be wild!
-  if (aFrameRate != mFrameRate) {
-    mFrameRate = aFrameRate;
-    mOMXReconfigure = true;  // force re-configure on next frame
-  }
-#endif
-
-  // XXX Limit bitrate for 8x10 devices to a specific level depending on fps and resolution
-  // mBitRateKbps = LimitBitrate8x10(mWidth, mHeight, mFrameRate, aBitRateKbps);
-  // Rely on global single setting (~720 kbps for HVGA@30fps) for now
-  if (aBitRateKbps > 700) {
-    aBitRateKbps = 700;
-  }
-  mBitRateKbps = aBitRateKbps;
-  nsresult rv = mOMX->SetBitrate(mBitRateKbps);
-  NS_WARNING_ASSERTION(NS_SUCCEEDED(rv), "SetBitrate failed");
-  return NS_FAILED(rv) ? WEBRTC_VIDEO_CODEC_OK : WEBRTC_VIDEO_CODEC_ERROR;
-}
-
-// Decoder.
-WebrtcOMXH264VideoDecoder::WebrtcOMXH264VideoDecoder()
-  : mCallback(nullptr)
-  , mOMX(nullptr)
-{
-  mReservation = new OMXCodecReservation(false);
-  CODEC_LOGD("WebrtcOMXH264VideoDecoder:%p will be constructed", this);
-}
-
-int32_t
-WebrtcOMXH264VideoDecoder::InitDecode(const webrtc::VideoCodec* aCodecSettings,
-                                      int32_t aNumOfCores)
-{
-  CODEC_LOGD("WebrtcOMXH264VideoDecoder:%p init OMX:%p", this, mOMX.get());
-
-  if (!mReservation->ReserveOMXCodec()) {
-    CODEC_LOGD("WebrtcOMXH264VideoDecoder:%p Decoder in use", this);
-    return WEBRTC_VIDEO_CODEC_ERROR;
-  }
-
-  // Defer configuration until SPS/PPS NALUs (where actual decoder config
-  // values can be extracted) are received.
-
-  CODEC_LOGD("WebrtcOMXH264VideoDecoder:%p OMX Decoder reserved", this);
-  return WEBRTC_VIDEO_CODEC_OK;
-}
-
-int32_t
-WebrtcOMXH264VideoDecoder::Decode(const webrtc::EncodedImage& aInputImage,
-                                  bool aMissingFrames,
-                                  const webrtc::RTPFragmentationHeader* aFragmentation,
-                                  const webrtc::CodecSpecificInfo* aCodecSpecificInfo,
-                                  int64_t aRenderTimeMs)
-{
-  if (aInputImage._length== 0 || !aInputImage._buffer) {
-    return WEBRTC_VIDEO_CODEC_ERROR;
-  }
-
-  bool configured = !!mOMX;
-  if (!configured) {
-    // Search for SPS NALU in input to get width/height config.
-    int32_t width;
-    int32_t height;
-    status_t result = WebrtcOMXDecoder::ExtractPicDimensions(aInputImage._buffer,
-                                                             aInputImage._length,
-                                                             &width, &height);
-    if (result != OK) {
-      // Cannot config decoder because SPS haven't been seen.
-      CODEC_LOGI("WebrtcOMXH264VideoDecoder:%p missing SPS in input (nal 0x%02x, len %d)",
-                 this, aInputImage._buffer[sizeof(kNALStartCode)] & 0x1f, aInputImage._length);
-      return WEBRTC_VIDEO_CODEC_UNINITIALIZED;
-    }
-    RefPtr<WebrtcOMXDecoder> omx = new WebrtcOMXDecoder(MEDIA_MIMETYPE_VIDEO_AVC,
-                                                        mCallback);
-    result = omx->ConfigureWithPicDimensions(width, height);
-    if (NS_WARN_IF(result != OK)) {
-      return WEBRTC_VIDEO_CODEC_UNINITIALIZED;
-    }
-    CODEC_LOGD("WebrtcOMXH264VideoDecoder:%p start OMX", this);
-    mOMX = omx;
-  }
-
-  bool feedFrame = true;
-  while (feedFrame) {
-    status_t err = mOMX->FillInput(aInputImage, !configured, aRenderTimeMs);
-    feedFrame = (err == -EAGAIN); // No input buffer available. Try again.
-  }
-
-  return WEBRTC_VIDEO_CODEC_OK;
-}
-
-int32_t
-WebrtcOMXH264VideoDecoder::RegisterDecodeCompleteCallback(webrtc::DecodedImageCallback* aCallback)
-{
-  CODEC_LOGD("WebrtcOMXH264VideoDecoder:%p set callback:%p", this, aCallback);
-  MOZ_ASSERT(aCallback);
-  mCallback = aCallback;
-
-  return WEBRTC_VIDEO_CODEC_OK;
-}
-
-int32_t
-WebrtcOMXH264VideoDecoder::Release()
-{
-  CODEC_LOGD("WebrtcOMXH264VideoDecoder:%p will be released", this);
-
-  mOMX = nullptr; // calls Stop()
-  mReservation->ReleaseOMXCodec();
-
-  return WEBRTC_VIDEO_CODEC_OK;
-}
-
-WebrtcOMXH264VideoDecoder::~WebrtcOMXH264VideoDecoder()
-{
-  CODEC_LOGD("WebrtcOMXH264VideoDecoder:%p will be destructed", this);
-  Release();
-}
-
-int32_t
-WebrtcOMXH264VideoDecoder::Reset()
-{
-  CODEC_LOGW("WebrtcOMXH264VideoDecoder::Reset() will NOT reset decoder");
-  return WEBRTC_VIDEO_CODEC_OK;
-}
-
-}
deleted file mode 100644
--- a/media/webrtc/signaling/src/media-conduit/WebrtcOMXH264VideoCodec.h
+++ /dev/null
@@ -1,108 +0,0 @@
-/* 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/. */
-
-#ifndef WEBRTC_GONK
-#pragma error WebrtcOMXH264VideoCodec works only on B2G.
-#endif
-
-#ifndef WEBRTC_OMX_H264_CODEC_H_
-#define WEBRTC_OMX_H264_CODEC_H_
-
-#include "AudioConduit.h"
-#include "VideoConduit.h"
-#include <foundation/ABase.h>
-#include <utils/RefBase.h>
-#include "OMXCodecWrapper.h"
-
-namespace android {
-  class OMXVideoEncoder;
-}
-
-namespace mozilla {
-
-class WebrtcOMXDecoder;
-class OMXOutputDrain;
-
-// XXX see if we can reduce this
-#define WEBRTC_OMX_H264_MIN_DECODE_BUFFERS 10
-#define OMX_IDR_NEEDED_FOR_BITRATE 0
-
-class WebrtcOMXH264VideoEncoder : public WebrtcVideoEncoder
-{
-public:
-  WebrtcOMXH264VideoEncoder();
-
-  virtual ~WebrtcOMXH264VideoEncoder();
-
-  // Implement VideoEncoder interface.
-  virtual uint64_t PluginID() const override { return 0; }
-
-  virtual int32_t InitEncode(const webrtc::VideoCodec* aCodecSettings,
-                             int32_t aNumOfCores,
-                             size_t aMaxPayloadSize) override;
-
-  virtual int32_t Encode(const webrtc::VideoFrame& aInputImage,
-                         const webrtc::CodecSpecificInfo* aCodecSpecificInfo,
-                         const std::vector<webrtc::FrameType>* aFrameTypes) override;
-
-  virtual int32_t RegisterEncodeCompleteCallback(webrtc::EncodedImageCallback* aCallback) override;
-
-  virtual int32_t Release() override;
-
-  virtual int32_t SetChannelParameters(uint32_t aPacketLossRate,
-                                       int64_t aRoundTripTimeMs) override;
-
-  virtual int32_t SetRates(uint32_t aBitRate, uint32_t aFrameRate) override;
-
-private:
-  nsAutoPtr<android::OMXVideoEncoder> mOMX;
-  android::sp<android::OMXCodecReservation> mReservation;
-
-  webrtc::EncodedImageCallback* mCallback;
-  RefPtr<OMXOutputDrain> mOutputDrain;
-  uint32_t mWidth;
-  uint32_t mHeight;
-  uint32_t mFrameRate;
-  uint32_t mBitRateKbps;
-#ifdef OMX_IDR_NEEDED_FOR_BITRATE
-  uint32_t mBitRateAtLastIDR;
-  TimeStamp mLastIDRTime;
-#endif
-  bool mOMXConfigured;
-  bool mOMXReconfigure;
-  webrtc::EncodedImage mEncodedImage;
-};
-
-class WebrtcOMXH264VideoDecoder : public WebrtcVideoDecoder
-{
-public:
-  WebrtcOMXH264VideoDecoder();
-
-  virtual ~WebrtcOMXH264VideoDecoder();
-
-  // Implement VideoDecoder interface.
-  virtual uint64_t PluginID() const override { return 0; }
-
-  virtual int32_t InitDecode(const webrtc::VideoCodec* aCodecSettings,
-                             int32_t aNumOfCores) override;
-  virtual int32_t Decode(const webrtc::EncodedImage& aInputImage,
-                         bool aMissingFrames,
-                         const webrtc::RTPFragmentationHeader* aFragmentation,
-                         const webrtc::CodecSpecificInfo* aCodecSpecificInfo = nullptr,
-                         int64_t aRenderTimeMs = -1) override;
-  virtual int32_t RegisterDecodeCompleteCallback(webrtc::DecodedImageCallback* callback) override;
-
-  virtual int32_t Release() override;
-
-  virtual int32_t Reset() override;
-
-private:
-  webrtc::DecodedImageCallback* mCallback;
-  RefPtr<WebrtcOMXDecoder> mOMX;
-  android::sp<android::OMXCodecReservation> mReservation;
-};
-
-}
-
-#endif // WEBRTC_OMX_H264_CODEC_H_
--- a/media/webrtc/signaling/src/mediapipeline/MediaPipeline.cpp
+++ b/media/webrtc/signaling/src/mediapipeline/MediaPipeline.cpp
@@ -2122,17 +2122,18 @@ void MediaPipelineReceiveAudio::DetachMe
 
     if (stream_->GraphImpl()) {
       stream_->RemoveListener(listener_);
     }
     stream_ = nullptr;
   }
 }
 
-nsresult MediaPipelineReceiveAudio::Init() {
+nsresult MediaPipelineReceiveAudio::Init()
+{
   ASSERT_ON_THREAD(main_thread_);
   MOZ_MTLOG(ML_DEBUG, __FUNCTION__);
 
   description_ = pc_ + "| Receive audio[";
   description_ += track_id_;
   description_ += "]";
 
   listener_->AddSelf();
@@ -2143,125 +2144,112 @@ nsresult MediaPipelineReceiveAudio::Init
 void MediaPipelineReceiveAudio::SetPrincipalHandle_m(const PrincipalHandle& principal_handle)
 {
   listener_->SetPrincipalHandle_m(principal_handle);
 }
 
 class MediaPipelineReceiveVideo::PipelineListener
   : public GenericReceiveListener {
 public:
-  PipelineListener(SourceMediaStream * source, TrackID track_id)
-    : GenericReceiveListener(source, track_id),
-      width_(0),
-      height_(0),
-      image_container_(),
-      image_(),
-      monitor_("Video PipelineListener")
+  PipelineListener(SourceMediaStream* source, TrackID track_id)
+    : GenericReceiveListener(source, track_id)
+    , image_container_()
+    , image_()
+    , mutex_("Video PipelineListener")
   {
     image_container_ =
       LayerManager::CreateImageContainer(ImageContainer::ASYNCHRONOUS);
   }
 
   // Implement MediaStreamListener
   void NotifyPull(MediaStreamGraph* graph, StreamTime desired_time) override
   {
-    ReentrantMonitorAutoEnter enter(monitor_);
+    MutexAutoLock lock(mutex_);
 
     RefPtr<Image> image = image_;
     StreamTime delta = desired_time - played_ticks_;
 
     // Don't append if we've already provided a frame that supposedly
     // goes past the current aDesiredTime Doing so means a negative
     // delta and thus messes up handling of the graph
     if (delta > 0) {
       VideoSegment segment;
-      segment.AppendFrame(image.forget(), delta, IntSize(width_, height_),
-                          principal_handle_);
+      IntSize size = image ? image->GetSize() : IntSize(width_, height_);
+      segment.AppendFrame(image.forget(), delta, size, principal_handle_);
       // Handle track not actually added yet or removed/finished
       if (source_->AppendToTrack(track_id_, &segment)) {
         played_ticks_ = desired_time;
       } else {
         MOZ_MTLOG(ML_ERROR, "AppendToTrack failed");
         return;
       }
     }
   }
 
   // Accessors for external writes from the renderer
   void FrameSizeChange(unsigned int width,
                        unsigned int height,
                        unsigned int number_of_streams) {
-    ReentrantMonitorAutoEnter enter(monitor_);
+    MutexAutoLock enter(mutex_);
 
     width_ = width;
     height_ = height;
   }
 
   void RenderVideoFrame(const webrtc::VideoFrameBuffer& buffer,
                         uint32_t time_stamp,
-                        int64_t render_time,
-                        const RefPtr<layers::Image>& video_image)
+                        int64_t render_time)
   {
-    RenderVideoFrame(buffer.DataY(),
-                     buffer.StrideY(),
-                     buffer.DataU(),
-                     buffer.StrideU(),
-                     buffer.DataV(),
-                     buffer.StrideV(),
-                     time_stamp, render_time, video_image);
-  }
+    if (buffer.native_handle()) {
+      // We assume that only native handles are used with the
+      // WebrtcMediaDataDecoderCodec decoder.
+      RefPtr<Image> image = static_cast<Image*>(buffer.native_handle());
+      MutexAutoLock lock(mutex_);
+      image_ = image;
+      return;
+    }
 
-  void RenderVideoFrame(const uint8_t* buffer_y,
-                        uint32_t y_stride,
-                        const uint8_t* buffer_u,
-                        uint32_t u_stride,
-                        const uint8_t* buffer_v,
-                        uint32_t v_stride,
-                        uint32_t time_stamp,
-                        int64_t render_time,
-                        const RefPtr<layers::Image>& video_image)
-  {
-    ReentrantMonitorAutoEnter enter(monitor_);
+    MOZ_ASSERT(buffer.DataY());
+    // Create a video frame using |buffer|.
+    RefPtr<PlanarYCbCrImage> yuvImage =
+      image_container_->CreatePlanarYCbCrImage();
 
-    if (buffer_y) {
-      // Create a video frame using |buffer|.
-      RefPtr<PlanarYCbCrImage> yuvImage = image_container_->CreatePlanarYCbCrImage();
+    PlanarYCbCrData yuvData;
+    yuvData.mYChannel = const_cast<uint8_t*>(buffer.DataY());
+    yuvData.mYSize = IntSize(buffer.width(), buffer.height());
+    yuvData.mYStride = buffer.StrideY();
+    MOZ_ASSERT(buffer.StrideU() == buffer.StrideV());
+    yuvData.mCbCrStride = buffer.StrideU();
+    yuvData.mCbChannel = const_cast<uint8_t*>(buffer.DataU());
+    yuvData.mCrChannel = const_cast<uint8_t*>(buffer.DataV());
+    yuvData.mCbCrSize =
+      IntSize((buffer.width() + 1) >> 1, (buffer.height() + 1) >> 1);
+    yuvData.mPicX = 0;
+    yuvData.mPicY = 0;
+    yuvData.mPicSize = IntSize(buffer.width(), buffer.height());
+    yuvData.mStereoMode = StereoMode::MONO;
 
-      PlanarYCbCrData yuvData;
-      yuvData.mYChannel = const_cast<uint8_t*>(buffer_y);
-      yuvData.mYSize = IntSize(y_stride, height_);
-      yuvData.mYStride = y_stride;
-      MOZ_ASSERT(u_stride == v_stride);
-      yuvData.mCbCrStride = u_stride;
-      yuvData.mCbChannel = const_cast<uint8_t*>(buffer_u);
-      yuvData.mCrChannel = const_cast<uint8_t*>(buffer_v);
-      yuvData.mCbCrSize = IntSize(yuvData.mCbCrStride, (height_ + 1) >> 1);
-      yuvData.mPicX = 0;
-      yuvData.mPicY = 0;
-      yuvData.mPicSize = IntSize(width_, height_);
-      yuvData.mStereoMode = StereoMode::MONO;
+    if (!yuvImage->CopyData(yuvData)) {
+      MOZ_ASSERT(false);
+      return;
+    }
 
-      if (!yuvImage->CopyData(yuvData)) {
-        MOZ_ASSERT(false);
-        return;
-      }
-
-      image_ = yuvImage;
-    }
+    MutexAutoLock lock(mutex_);
+    image_ = yuvImage;
   }
 
 private:
   int width_;
   int height_;
   RefPtr<layers::ImageContainer> image_container_;
   RefPtr<layers::Image> image_;
-  mozilla::ReentrantMonitor monitor_; // Monitor for processing WebRTC frames.
-                                      // Protects image_ against:
-                                      // - Writing from the GIPS thread
-                                      // - Reading from the MSG thread
+  Mutex mutex_; // Mutex for processing WebRTC frames.
+                // Protects image_ against:
+                // - Writing from the GIPS thread
+                // - Reading from the MSG thread
 };
 
 class MediaPipelineReceiveVideo::PipelineRenderer : public mozilla::VideoRenderer
 {
 public:
   explicit PipelineRenderer(MediaPipelineReceiveVideo *pipeline) :
     pipeline_(pipeline) {}
 
@@ -2272,39 +2260,19 @@ public:
                        unsigned int height,
                        unsigned int number_of_streams) override
   {
     pipeline_->listener_->FrameSizeChange(width, height, number_of_streams);
   }
 
   void RenderVideoFrame(const webrtc::VideoFrameBuffer& buffer,
                         uint32_t time_stamp,
-                        int64_t render_time,
-                        const ImageHandle& handle) override
+                        int64_t render_time) override
   {
-    pipeline_->listener_->RenderVideoFrame(buffer,
-                                           time_stamp, render_time,
-                                           handle.GetImage());
-  }
-
-  void RenderVideoFrame(const uint8_t* buffer_y,
-                        uint32_t y_stride,
-                        const uint8_t* buffer_u,
-                        uint32_t u_stride,
-                        const uint8_t* buffer_v,
-                        uint32_t v_stride,
-                        uint32_t time_stamp,
-                        int64_t render_time,
-                        const ImageHandle& handle) override
-  {
-    pipeline_->listener_->RenderVideoFrame(buffer_y, y_stride,
-                                           buffer_u, u_stride,
-                                           buffer_v, v_stride,
-                                           time_stamp, render_time,
-                                           handle.GetImage());
+    pipeline_->listener_->RenderVideoFrame(buffer, time_stamp, render_time);
   }
 
 private:
   MediaPipelineReceiveVideo *pipeline_;  // Raw pointer to avoid cycles
 };
 
 
 MediaPipelineReceiveVideo::MediaPipelineReceiveVideo(
--- a/mfbt/Assertions.h
+++ b/mfbt/Assertions.h
@@ -273,16 +273,20 @@ static MOZ_COLD MOZ_NORETURN MOZ_NEVER_I
 
 /*
  * MOZ_CRASH_UNSAFE_OOL(explanation-string) can be used if the explanation
  * string cannot be a string literal (but no other processing needs to be done
  * on it). A regular MOZ_CRASH() is preferred wherever possible, as passing
  * arbitrary strings from a potentially compromised process is not without risk.
  * If the string being passed is the result of a printf-style function,
  * consider using MOZ_CRASH_UNSAFE_PRINTF instead.
+ *
+ * @note This macro causes data collection because crash strings are annotated
+ * to crash-stats and are publicly visible. Firefox data stewards must do data
+ * review on usages of this macro.
  */
 #ifndef DEBUG
 MFBT_API MOZ_COLD MOZ_NORETURN MOZ_NEVER_INLINE void
 MOZ_CrashOOL(int aLine, const char* aReason);
 #  define MOZ_CRASH_UNSAFE_OOL(reason) MOZ_CrashOOL(__LINE__, reason)
 #else
 MFBT_API MOZ_COLD MOZ_NORETURN MOZ_NEVER_INLINE void
 MOZ_CrashOOL(const char* aFilename, int aLine, const char* aReason);
@@ -306,16 +310,20 @@ MOZ_CrashPrintf(const char* aFilename, i
 
 /*
  * MOZ_CRASH_UNSAFE_PRINTF(format, arg1 [, args]) can be used when more
  * information is desired than a string literal can supply. The caller provides
  * a printf-style format string, which must be a string literal and between
  * 1 and 4 additional arguments. A regular MOZ_CRASH() is preferred wherever
  * possible, as passing arbitrary strings to printf from a potentially
  * compromised process is not without risk.
+ *
+ * @note This macro causes data collection because crash strings are annotated
+ * to crash-stats and are publicly visible. Firefox data stewards must do data
+ * review on usages of this macro.
  */
 #define MOZ_CRASH_UNSAFE_PRINTF(format, ...) \
    do { \
      static_assert( \
        MOZ_ARG_COUNT(__VA_ARGS__) > 0, \
        "Did you forget arguments to MOZ_CRASH_UNSAFE_PRINTF? " \
        "Or maybe you want MOZ_CRASH instead?"); \
      static_assert( \
--- a/mobile/android/components/extensions/ext-utils.js
+++ b/mobile/android/components/extensions/ext-utils.js
@@ -393,16 +393,20 @@ class Tab extends TabBase {
   get index() {
     return this.window.BrowserApp.tabs.indexOf(this.nativeTab);
   }
 
   get mutedInfo() {
     return {muted: false};
   }
 
+  get lastAccessed() {
+    return this.nativeTab.lastTouchedAt;
+  }
+
   get pinned() {
     return false;
   }
 
   get active() {
     return this.nativeTab.getActive();
   }
 
--- a/mobile/android/components/extensions/schemas/tabs.json
+++ b/mobile/android/components/extensions/schemas/tabs.json
@@ -59,16 +59,17 @@
           "id": {"type": "integer", "minimum": -1, "optional": true, "description": "The ID of the tab. Tab IDs are unique within a browser session. Under some circumstances a Tab may not be assigned an ID, for example when querying foreign tabs using the $(ref:sessions) API, in which case a session ID may be present. Tab ID can also be set to $(ref:tabs.TAB_ID_NONE) for apps and devtools windows."},
           "index": {"type": "integer", "minimum": -1, "description": "The zero-based index of the tab within its window."},
           "windowId": {"type": "integer", "minimum": 0, "description": "The ID of the window the tab is contained within."},
           "openerTabId": {"unsupported": true, "type": "integer", "minimum": 0, "optional": true, "description": "The ID of the tab that opened this tab, if any. This property is only present if the opener tab still exists."},
           "selected": {"type": "boolean", "description": "Whether the tab is selected.", "deprecated": "Please use $(ref:tabs.Tab.highlighted).", "unsupported": true},
           "highlighted": {"type": "boolean", "description": "Whether the tab is highlighted. Works as an alias of active."},
           "active": {"type": "boolean", "description": "Whether the tab is active in its window. (Does not necessarily mean the window is focused.)"},
           "pinned": {"type": "boolean", "description": "Whether the tab is pinned."},
+          "lastAccessed": {"type": "integer", "optional": true, "description": "The last time the tab was accessed as the number of milliseconds since epoch."},
           "audible": {"type": "boolean", "optional": true, "description": "Whether the tab has produced sound over the past couple of seconds (but it might not be heard if also muted). Equivalent to whether the speaker audio indicator is showing."},
           "mutedInfo": {"$ref": "MutedInfo", "optional": true, "description": "Current tab muted state and the reason for the last state change."},
           "url": {"type": "string", "optional": true, "permissions": ["tabs"], "description": "The URL the tab is displaying. This property is only present if the extension's manifest includes the <code>\"tabs\"</code> permission."},
           "title": {"type": "string", "optional": true, "permissions": ["tabs"], "description": "The title of the tab. This property is only present if the extension's manifest includes the <code>\"tabs\"</code> permission."},
           "favIconUrl": {"type": "string", "optional": true, "permissions": ["tabs"], "description": "The URL of the tab's favicon. This property is only present if the extension's manifest includes the <code>\"tabs\"</code> permission. It may also be an empty string if the tab is loading."},
           "status": {"type": "string", "optional": true, "description": "Either <em>loading</em> or <em>complete</em>."},
           "incognito": {"type": "boolean", "description": "Whether the tab is in an incognito window."},
           "width": {"type": "integer", "optional": true, "description": "The width of the tab in pixels."},
--- a/mobile/android/components/extensions/test/mochitest/mochitest.ini
+++ b/mobile/android/components/extensions/test/mochitest/mochitest.ini
@@ -21,13 +21,14 @@ tags = webextensions
 [test_ext_tabs_executeScript.html]
 [test_ext_tabs_executeScript_bad.html]
 skip-if = true # Currently fails in emulator runs
 [test_ext_tabs_executeScript_good.html]
 [test_ext_tabs_executeScript_no_create.html]
 [test_ext_tabs_executeScript_runAt.html]
 [test_ext_tabs_getCurrent.html]
 [test_ext_tabs_insertCSS.html]
+[test_ext_tabs_lastAccessed.html]
 [test_ext_tabs_reload.html]
 [test_ext_tabs_reload_bypass_cache.html]
 [test_ext_tabs_onUpdated.html]
 [test_ext_tabs_sendMessage.html]
 [test_ext_tabs_update_url.html]
new file mode 100644
--- /dev/null
+++ b/mobile/android/components/extensions/test/mochitest/test_ext_tabs_lastAccessed.html
@@ -0,0 +1,58 @@
+<!DOCTYPE HTML>
+<html>
+<head>
+  <title>Tabs lastAccessed Test</title>
+  <script type="text/javascript" src="/tests/SimpleTest/SimpleTest.js"></script>
+  <script type="text/javascript" src="/tests/SimpleTest/SpawnTask.js"></script>
+  <script type="text/javascript" src="/tests/SimpleTest/ExtensionTestUtils.js"></script>
+  <script type="text/javascript" src="head.js"></script>
+  <link rel="stylesheet" href="/tests/SimpleTest/test.css"/>
+</head>
+<body>
+
+<script type="text/javascript">
+"use strict";
+
+add_task(async function testLastAccessed() {
+  let past = Date.now();
+
+  window.open("https://example.com/?1");
+  window.open("https://example.com/?2");
+
+  let extension = ExtensionTestUtils.loadExtension({
+    manifest: {
+      permissions: ["tabs"],
+    },
+    async background() {
+      browser.test.onMessage.addListener(async function(msg, past) {
+        if (msg !== "past") {
+          return;
+        }
+
+        let [tab1] = await browser.tabs.query({url: "https://example.com/?1"});
+        let [tab2] = await browser.tabs.query({url: "https://example.com/?2"});
+
+        browser.test.assertTrue(tab1 && tab2, "Expected tabs were found");
+
+        let now = Date.now();
+
+        browser.test.assertTrue(past < tab1.lastAccessed &&
+                                tab1.lastAccessed < tab2.lastAccessed &&
+                                tab2.lastAccessed <= now,
+                                "lastAccessed timestamps are recent and in the right order");
+
+        await browser.tabs.remove([tab1.id, tab2.id]);
+
+        browser.test.notifyPass("tabs.lastAccessed");
+      });
+    },
+  });
+
+  await extension.startup();
+  await extension.sendMessage("past", past);
+  await extension.awaitFinish("tabs.lastAccessed");
+  await extension.unload();
+});
+</script>
+
+</body>
--- a/modules/libpref/init/all.js
+++ b/modules/libpref/init/all.js
@@ -558,16 +558,19 @@ pref("media.peerconnection.capture_delay
 pref("media.getusermedia.playout_delay", 50);
 pref("media.navigator.audio.full_duplex", true);
 #else
 // *BSD, others - merely a guess for now
 pref("media.peerconnection.capture_delay", 50);
 pref("media.getusermedia.playout_delay", 50);
 pref("media.navigator.audio.full_duplex", false);
 #endif
+// Use MediaDataDecoder API for WebRTC, this includes hardware acceleration for
+// decoding.
+pref("media.navigator.mediadatadecoder_enabled", false);
 #endif
 
 pref("dom.webaudio.enabled", true);
 
 #if !defined(ANDROID)
 pref("media.getusermedia.screensharing.enabled", true);
 #endif
 
--- a/npm-shrinkwrap.json
+++ b/npm-shrinkwrap.json
@@ -175,17 +175,17 @@
       "resolved": "https://registry.npmjs.org/core-util-is/-/core-util-is-1.0.2.tgz",
       "integrity": "sha1-tf1UIgqivFq1eqtxQMlAdUUDwac="
     },
     "d": {
       "version": "1.0.0",
       "resolved": "https://registry.npmjs.org/d/-/d-1.0.0.tgz",
       "integrity": "sha1-dUu1v+VUUdpppYuU1F9MWwRi1Y8=",
       "requires": {
-        "es5-ext": "0.10.23"
+        "es5-ext": "0.10.24"
       }
     },
     "debug": {
       "version": "2.6.8",
       "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.8.tgz",
       "integrity": "sha1-5zFTHKLt4n0YgiJCfaF4IdaP9Pw=",
       "requires": {
         "ms": "2.0.0"
@@ -258,75 +258,75 @@
       }
     },
     "entities": {
       "version": "1.1.1",
       "resolved": "https://registry.npmjs.org/entities/-/entities-1.1.1.tgz",
       "integrity": "sha1-blwtClYhtdra7O+AuQ7ftc13cvA="
     },
     "es5-ext": {
-      "version": "0.10.23",
-      "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.23.tgz",
-      "integrity": "sha1-dXi1G+l0IHpUh4IbVlOMIk5Oezg=",
+      "version": "0.10.24",
+      "resolved": "https://registry.npmjs.org/es5-ext/-/es5-ext-0.10.24.tgz",
+      "integrity": "sha1-pVh3yZJLwMjZvTwsvhdJWsFwmxQ=",
       "requires": {
         "es6-iterator": "2.0.1",
         "es6-symbol": "3.1.1"
       }
     },
     "es6-iterator": {
       "version": "2.0.1",
       "resolved": "https://registry.npmjs.org/es6-iterator/-/es6-iterator-2.0.1.tgz",
       "integrity": "sha1-jjGcnwRTv1ddN0lAplWSDlnKVRI=",
       "requires": {
         "d": "1.0.0",
-        "es5-ext": "0.10.23",
+        "es5-ext": "0.10.24",
         "es6-symbol": "3.1.1"
       }
     },
     "es6-map": {
       "version": "0.1.5",
       "resolved": "https://registry.npmjs.org/es6-map/-/es6-map-0.1.5.tgz",
       "integrity": "sha1-kTbgUD3MBqMBaQ8LsU/042TpSfA=",
       "requires": {
         "d": "1.0.0",
-        "es5-ext": "0.10.23",
+        "es5-ext": "0.10.24",
         "es6-iterator": "2.0.1",
         "es6-set": "0.1.5",
         "es6-symbol": "3.1.1",
         "event-emitter": "0.3.5"
       }
     },
     "es6-set": {
       "version": "0.1.5",
       "resolved": "https://registry.npmjs.org/es6-set/-/es6-set-0.1.5.tgz",
       "integrity": "sha1-0rPsXU2ADO2BjbU40ol02wpzzLE=",
       "requires": {
         "d": "1.0.0",
-        "es5-ext": "0.10.23",
+        "es5-ext": "0.10.24",
         "es6-iterator": "2.0.1",
         "es6-symbol": "3.1.1",
         "event-emitter": "0.3.5"
       }
     },
     "es6-symbol": {
       "version": "3.1.1",
       "resolved": "https://registry.npmjs.org/es6-symbol/-/es6-symbol-3.1.1.tgz",
       "integrity": "sha1-vwDvT9q2uhtG7Le2KbTH7VcVzHc=",
       "requires": {
         "d": "1.0.0",
-        "es5-ext": "0.10.23"
+        "es5-ext": "0.10.24"
       }
     },
     "es6-weak-map": {
       "version": "2.0.2",
       "resolved": "https://registry.npmjs.org/es6-weak-map/-/es6-weak-map-2.0.2.tgz",
       "integrity": "sha1-XjqzIlH/0VOKH45f+hNXdy+S2W8=",
       "requires": {
         "d": "1.0.0",
-        "es5-ext": "0.10.23",
+        "es5-ext": "0.10.24",
         "es6-iterator": "2.0.1",
         "es6-symbol": "3.1.1"
       }
     },
     "escape-string-regexp": {
       "version": "1.0.5",
       "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
       "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ="
@@ -360,17 +360,17 @@
         "file-entry-cache": "2.0.0",
         "glob": "7.1.2",
         "globals": "9.18.0",
         "ignore": "3.3.3",
         "imurmurhash": "0.1.4",
         "inquirer": "0.12.0",
         "is-my-json-valid": "2.16.0",
         "is-resolvable": "1.0.0",
-        "js-yaml": "3.8.4",
+        "js-yaml": "3.9.0",
         "json-stable-stringify": "1.0.1",
         "levn": "0.3.0",
         "lodash": "4.17.4",
         "mkdirp": "0.5.1",
         "natural-compare": "1.4.0",
         "optionator": "0.8.2",
         "path-is-inside": "1.0.2",
         "pluralize": "1.2.1",
@@ -429,19 +429,19 @@
       "resolved": "https://registry.npmjs.org/espree/-/espree-3.4.3.tgz",
       "integrity": "sha1-KRC1zNSc6JPC//+qtP2LOjG4I3Q=",
       "requires": {
         "acorn": "5.1.1",
         "acorn-jsx": "3.0.1"
       }
     },
     "esprima": {
-      "version": "3.1.3",
-      "resolved": "https://registry.npmjs.org/esprima/-/esprima-3.1.3.tgz",
-      "integrity": "sha1-/cpRzuYTOJXjyI1TXOSdv/YqRjM="
+      "version": "4.0.0",
+      "resolved": "https://registry.npmjs.org/esprima/-/esprima-4.0.0.tgz",
+      "integrity": "sha512-oftTcaMu/EGrEIu904mWteKIv8vMuOgGYo7EhVJJN00R/EED9DCua/xxHRdYnKtcECzVg7xOWhflvJMnqcFZjw=="
     },
     "esquery": {
       "version": "1.0.0",
       "resolved": "https://registry.npmjs.org/esquery/-/esquery-1.0.0.tgz",
       "integrity": "sha1-z7qLV9f7qT8XKYqKAGoEzaE9gPo=",
       "requires": {
         "estraverse": "4.2.0"
       }
@@ -466,17 +466,17 @@
       "integrity": "sha1-Cr9PHKpbyx96nYrMbepPqqBLrJs="
     },
     "event-emitter": {
       "version": "0.3.5",
       "resolved": "https://registry.npmjs.org/event-emitter/-/event-emitter-0.3.5.tgz",
       "integrity": "sha1-34xp7vFkeSPHFXuc6DhAYQsCzDk=",
       "requires": {
         "d": "1.0.0",
-        "es5-ext": "0.10.23"
+        "es5-ext": "0.10.24"
       }
     },
     "exit-hook": {
       "version": "1.1.1",
       "resolved": "https://registry.npmjs.org/exit-hook/-/exit-hook-1.1.1.tgz",
       "integrity": "sha1-8FyiM7SMBdVP/wd2XfhQfpXAL/g="
     },
     "fast-levenshtein": {
@@ -714,22 +714,22 @@
       "integrity": "sha1-u5NdSFgsuhaMBoNJV6VKPgcSTxE="
     },
     "js-tokens": {
       "version": "3.0.2",
       "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-3.0.2.tgz",
       "integrity": "sha1-mGbfOVECEw449/mWvOtlRDIJwls="
     },
     "js-yaml": {
-      "version": "3.8.4",
-      "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.8.4.tgz",
-      "integrity": "sha1-UgtFZPhlc7qWZir4Woyvp7S1pvY=",
+      "version": "3.9.0",
+      "resolved": "https://registry.npmjs.org/js-yaml/-/js-yaml-3.9.0.tgz",
+      "integrity": "sha512-0LoUNELX4S+iofCT8f4uEHIiRBR+c2AINyC8qRWfC6QNruLtxVZRJaPcu/xwMgFIgDxF25tGHaDjvxzJCNE9yw==",
       "requires": {
         "argparse": "1.0.9",
-        "esprima": "3.1.3"
+        "esprima": "4.0.0"
       }
     },
     "json-stable-stringify": {
       "version": "1.0.1",
       "resolved": "https://registry.npmjs.org/json-stable-stringify/-/json-stable-stringify-1.0.1.tgz",
       "integrity": "sha1-mnWdOcXy/1A/1TAGRu1EX4jE+a8=",
       "requires": {
         "jsonify": "0.0.0"
--- a/package.json
+++ b/package.json
@@ -1,19 +1,14 @@
 {
   "name": "mozillaeslintsetup",
   "description": "This package file is for setup of ESLint only for editor integration.",
   "repository": {},
   "license": "MPL-2.0",
   "dependencies": {
-    "escope": "^3.6.0",
     "eslint": "3.19.0",
     "eslint-plugin-html": "2.0.3",
     "eslint-plugin-mozilla": "file:tools/lint/eslint/eslint-plugin-mozilla",
     "eslint-plugin-no-unsanitized": "2.0.1",
     "eslint-plugin-react": "7.1.0",
-    "eslint-plugin-spidermonkey-js": "file:tools/lint/eslint/eslint-plugin-spidermonkey-js",
-    "espree": "^3.4.0",
-    "estraverse": "^4.2.0",
-    "ini-parser": "^0.0.2",
-    "sax": "^1.2.2"
+    "eslint-plugin-spidermonkey-js": "file:tools/lint/eslint/eslint-plugin-spidermonkey-js"
   }
 }
--- a/servo/Cargo.lock
+++ b/servo/Cargo.lock
@@ -1136,16 +1136,17 @@ source = "registry+https://github.com/ru
 name = "glutin_app"
 version = "0.0.1"
 dependencies = [
  "bitflags 0.7.0 (registry+https://github.com/rust-lang/crates.io-index)",
  "compositing 0.0.1",
  "euclid 0.15.1 (registry+https://github.com/rust-lang/crates.io-index)",
  "gdi32-sys 0.2.0 (registry+https://github.com/rust-lang/crates.io-index)",
  "gleam 0.4.7 (registry+https://github.com/rust-lang/crates.io-index)",
+ "libservo 0.0.1",
  "log 0.3.8 (registry+https://github.com/rust-lang/crates.io-index)",
  "msg 0.0.1",
  "net_traits 0.0.1",
  "osmesa-src 12.0.1 (git+https://github.com/servo/osmesa-src)",
  "osmesa-sys 0.1.2 (registry+https://github.com/rust-lang/crates.io-index)",
  "script_traits 0.0.1",
  "servo-egl 0.2.1 (registry+https://github.com/rust-lang/crates.io-index)",
  "servo-glutin 0.11.0 (registry+https://github.com/rust-lang/crates.io-index)",
--- a/servo/components/compositing/compositor.rs
+++ b/servo/components/compositing/compositor.rs
@@ -504,20 +504,17 @@ impl<Window: WindowMethods> IOCompositor
 
                 // Inform the embedder that the load has finished.
                 //
                 // TODO(pcwalton): Specify which frame's load completed.
                 self.window.load_end();
             }
 
             (Msg::AllowNavigation(url, response_chan), ShutdownState::NotShuttingDown) => {
-                let allow = self.window.allow_navigation(url);
-                if let Err(e) = response_chan.send(allow) {
-                    warn!("Failed to send allow_navigation result ({}).", e);
-                }
+                self.window.allow_navigation(url, response_chan);
             }
 
             (Msg::Recomposite(reason), ShutdownState::NotShuttingDown) => {
                 self.composition_request = CompositionRequest::CompositeNow(reason)
             }
 
             (Msg::KeyEvent(ch, key, state, modified), ShutdownState::NotShuttingDown) => {
                 if state == KeyState::Pressed {
--- a/servo/components/compositing/windowing.rs
+++ b/servo/components/compositing/windowing.rs
@@ -3,16 +3,17 @@
  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 
 //! Abstract windowing methods. The concrete implementations of these can be found in `platform/`.
 
 use compositor_thread::EventLoopWaker;
 use euclid::{Point2D, Size2D};
 use euclid::{TypedPoint2D, TypedRect, ScaleFactor, TypedSize2D};
 use gleam::gl;
+use ipc_channel::ipc::IpcSender;
 use msg::constellation_msg::{Key, KeyModifiers, KeyState, TraversalDirection};
 use net_traits::net_error_list::NetError;
 use script_traits::{DevicePixel, LoadData, MouseButton, TouchEventType, TouchId, TouchpadPressurePhase};
 use servo_geometry::DeviceIndependentPixel;
 use servo_url::ServoUrl;
 use std::fmt::{Debug, Error, Formatter};
 use std::rc::Rc;
 use style_traits::cursor::Cursor;
@@ -126,17 +127,17 @@ pub trait WindowMethods {
     fn status(&self, Option<String>);
     /// Called when the browser has started loading a frame.
     fn load_start(&self);
     /// Called when the browser is done loading a frame.
     fn load_end(&self);
     /// Called when the browser encounters an error while loading a URL
     fn load_error(&self, code: NetError, url: String);
     /// Wether or not to follow a link
-    fn allow_navigation(&self, url: ServoUrl) -> bool;
+    fn allow_navigation(&self, url: ServoUrl, IpcSender<bool>);
     /// Called when the <head> tag has finished parsing
     fn head_parsed(&self);
     /// Called when the history state has changed.
     fn history_changed(&self, Vec<LoadData>, usize);
 
     /// Returns the scale factor of the system (device pixels / device independent pixels).
     fn hidpi_factor(&self) -> ScaleFactor<f32, DeviceIndependentPixel, DevicePixel>;
 
--- a/servo/components/gfx/font_cache_thread.rs
+++ b/servo/components/gfx/font_cache_thread.rs
@@ -211,17 +211,18 @@ impl FontCache {
                     Some(url) => url.clone(),
                     None => return,
                 };
 
                 let request = RequestInit {
                     url: url.clone(),
                     type_: RequestType::Font,
                     destination: Destination::Font,
-                    origin: url.clone(),
+                    // TODO: Add a proper origin - Can't import GlobalScope from gfx
+                    // We can leave origin to be set by default
                     .. RequestInit::default()
                 };
 
                 let channel_to_self = self.channel_to_self.clone();
                 let bytes = Mutex::new(Vec::new());
                 let response_valid = Mutex::new(false);
                 debug!("Loading @font-face {} from {}", family_name, url);
                 fetch_async(request, &self.core_resource_thread, move |response| {
--- a/servo/components/layout/display_list_builder.rs
+++ b/servo/components/layout/display_list_builder.rs
@@ -51,27 +51,26 @@ use style::computed_values::{background_
 use style::computed_values::{background_repeat, border_style, cursor};
 use style::computed_values::{image_rendering, overflow_x, pointer_events, position, visibility};
 use style::logical_geometry::{LogicalPoint, LogicalRect, LogicalSize, WritingMode};
 use style::properties::{self, ServoComputedValues};
 use style::properties::longhands::border_image_repeat::computed_value::RepeatKeyword;
 use style::properties::style_structs;
 use style::servo::restyle_damage::REPAINT;
 use style::values::{Either, RGBA};
-use style::values::computed::{Angle, Gradient, GradientItem, LengthOrPercentage};
+use style::values::computed::{Angle, Gradient, GradientItem, LengthOrPercentage, Percentage};
 use style::values::computed::{LengthOrPercentageOrAuto, NumberOrPercentage, Position};
 use style::values::computed::effects::SimpleShadow;
 use style::values::computed::image::{EndingShape, LineDirection};
 use style::values::generics::background::BackgroundSize;
 use style::values::generics::effects::Filter;
 use style::values::generics::image::{Circle, Ellipse, EndingShape as GenericEndingShape};
 use style::values::generics::image::{GradientItem as GenericGradientItem, GradientKind};
 use style::values::generics::image::{Image, ShapeExtent};
 use style::values::generics::image::PaintWorklet;
-use style::values::specified::length::Percentage;
 use style::values::specified::position::{X, Y};
 use style_traits::CSSPixel;
 use style_traits::cursor::Cursor;
 use table_cell::CollapsedBordersForCell;
 use webrender_api::{ClipId, ColorF, ComplexClipRegion, GradientStop, LocalClip, RepeatMode};
 use webrender_api::{ScrollPolicy, TransformStyle};
 use webrender_helpers::{ToBorderRadius, ToMixBlendMode, ToRectF, ToTransformStyle};
 
--- a/servo/components/layout_thread/lib.rs
+++ b/servo/components/layout_thread/lib.rs
@@ -128,17 +128,16 @@ use style::selector_parser::SnapshotMap;
 use style::servo::restyle_damage::{REFLOW, REFLOW_OUT_OF_FLOW, REPAINT, REPOSITION, STORE_OVERFLOW};
 use style::shared_lock::{SharedRwLock, SharedRwLockReadGuard, StylesheetGuards};
 use style::stylearc::Arc as StyleArc;
 use style::stylesheets::{Origin, Stylesheet, StylesheetInDocument, UserAgentStylesheets};
 use style::stylist::{ExtraStyleData, Stylist};
 use style::thread_state;
 use style::timer::Timer;
 use style::traversal::{DomTraversal, TraversalDriver, TraversalFlags};
-use style::values::CompactCowStr;
 
 /// Information needed by the layout thread.
 pub struct LayoutThread {
     /// The ID of the pipeline that we belong to.
     id: PipelineId,
 
     /// The ID of the top-level browsing context that we belong to.
     top_level_browsing_context_id: TopLevelBrowsingContextId,
@@ -709,17 +708,17 @@ impl LayoutThread {
                 self.create_layout_thread(info)
             }
             Msg::SetFinalUrl(final_url) => {
                 self.url = final_url;
             },
             Msg::RegisterPaint(name, mut properties, painter) => {
                 debug!("Registering the painter");
                 let properties = properties.drain(..)
-                    .filter_map(|name| PropertyId::parse(CompactCowStr::from(&*name)).ok().map(|id| (name.clone(), id)))
+                    .filter_map(|name| PropertyId::parse(&*name).ok().map(|id| (name.clone(), id)))
                     .filter(|&(_, ref id)| id.as_shorthand().is_err())
                     .collect();
                 let registered_painter = RegisteredPainter {
                     name: name.clone(),
                     properties: properties,
                     painter: painter,
                 };
                 self.registered_painters.write()
--- a/servo/components/net/http_loader.rs
+++ b/servo/components/net/http_loader.rs
@@ -1366,17 +1366,17 @@ fn cors_check(request: &Request, respons
     // Step 4
     let origin = match origin {
         AccessControlAllowOrigin::Value(origin) => origin,
         // if it's Any or Null at this point, there's nothing to do but return Err(())
         _ => return Err(())
     };
 
     match request.origin {
-        Origin::Origin(ref o) if o.ascii_serialization() == origin => {},
+        Origin::Origin(ref o) if o.ascii_serialization() == origin.trim() => {},
         _ => return Err(())
     }
 
     // Step 5
     if request.credentials_mode != CredentialsMode::Include {
         return Ok(());
     }
 
--- a/servo/components/net_traits/request.rs
+++ b/servo/components/net_traits/request.rs
@@ -153,19 +153,17 @@ pub struct RequestInit {
     pub type_: Type,
     pub destination: Destination,
     pub synchronous: bool,
     pub mode: RequestMode,
     pub cache_mode: CacheMode,
     pub use_cors_preflight: bool,
     pub credentials_mode: CredentialsMode,
     pub use_url_credentials: bool,
-    // this should actually be set by fetch, but fetch
-    // doesn't have info about the client right now
-    pub origin: ServoUrl,
+    pub origin: ImmutableOrigin,
     // XXXManishearth these should be part of the client object
     pub referrer_url: Option<ServoUrl>,
     pub referrer_policy: Option<ReferrerPolicy>,
     pub pipeline_id: Option<PipelineId>,
     pub redirect_mode: RedirectMode,
     pub integrity_metadata: String,
     // to keep track of redirects
     pub url_list: Vec<ServoUrl>,
@@ -183,17 +181,17 @@ impl Default for RequestInit {
             type_: Type::None,
             destination: Destination::None,
             synchronous: false,
             mode: RequestMode::NoCors,
             cache_mode: CacheMode::Default,
             use_cors_preflight: false,
             credentials_mode: CredentialsMode::Omit,
             use_url_credentials: false,
-            origin: ServoUrl::parse("about:blank").unwrap(),
+            origin: ImmutableOrigin::new_opaque(),
             referrer_url: None,
             referrer_policy: None,
             pipeline_id: None,
             redirect_mode: RedirectMode::Follow,
             integrity_metadata: "".to_owned(),
             url_list: vec![],
         }
     }
@@ -297,17 +295,17 @@ impl Request {
             url_list: vec![url],
             redirect_count: 0,
             response_tainting: ResponseTainting::Basic,
         }
     }
 
     pub fn from_init(init: RequestInit) -> Request {
         let mut req = Request::new(init.url.clone(),
-                                   Some(Origin::Origin(init.origin.origin())),
+                                   Some(Origin::Origin(init.origin)),
                                    init.pipeline_id);
         req.method = init.method;
         req.headers = init.headers;
         req.unsafe_request = init.unsafe_request;
         req.body = init.body;
         req.service_workers_mode = init.service_workers_mode;
         req.type_ = init.type_;
         req.destination = init.destination;
--- a/servo/components/script/dom/cssstyledeclaration.rs
+++ b/servo/components/script/dom/cssstyledeclaration.rs
@@ -291,28 +291,28 @@ impl CSSStyleDeclarationMethods for CSSS
 
     // https://dev.w3.org/csswg/cssom/#dom-cssstyledeclaration-item
     fn Item(&self, index: u32) -> DOMString {
         self.IndexedGetter(index).unwrap_or_default()
     }
 
     // https://dev.w3.org/csswg/cssom/#dom-cssstyledeclaration-getpropertyvalue
     fn GetPropertyValue(&self, property: DOMString) -> DOMString {
-        let id = if let Ok(id) = PropertyId::parse(property.into()) {
+        let id = if let Ok(id) = PropertyId::parse(&property) {
             id
         } else {
             // Unkwown property
             return DOMString::new()
         };
         self.get_property_value(id)
     }
 
     // https://dev.w3.org/csswg/cssom/#dom-cssstyledeclaration-getpropertypriority
     fn GetPropertyPriority(&self, property: DOMString) -> DOMString {
-        let id = if let Ok(id) = PropertyId::parse(property.into()) {
+        let id = if let Ok(id) = PropertyId::parse(&property) {
             id
         } else {
             // Unkwown property
             return DOMString::new()
         };
 
         self.owner.with_block(|pdb| {
             if pdb.property_priority(&id).important() {
@@ -326,34 +326,34 @@ impl CSSStyleDeclarationMethods for CSSS
 
     // https://dev.w3.org/csswg/cssom/#dom-cssstyledeclaration-setproperty
     fn SetProperty(&self,
                    property: DOMString,
                    value: DOMString,
                    priority: DOMString)
                    -> ErrorResult {
         // Step 3
-        let id = if let Ok(id) = PropertyId::parse(property.into()) {
+        let id = if let Ok(id) = PropertyId::parse(&property) {
             id
         } else {
             // Unknown property
             return Ok(())
         };
         self.set_property(id, value, priority)
     }
 
     // https://dev.w3.org/csswg/cssom/#dom-cssstyledeclaration-setpropertypriority
     fn SetPropertyPriority(&self, property: DOMString, priority: DOMString) -> ErrorResult {
         // Step 1
         if self.readonly {
             return Err(Error::NoModificationAllowed);
         }
 
         // Step 2 & 3
-        let id = match PropertyId::parse(property.into()) {
+        let id = match PropertyId::parse(&property) {
             Ok(id) => id,
             Err(..) => return Ok(()), // Unkwown property
         };
 
         // Step 4
         let importance = match &*priority {
             "" => Importance::Normal,
             p if p.eq_ignore_ascii_case("important") => Importance::Important,
@@ -375,17 +375,17 @@ impl CSSStyleDeclarationMethods for CSSS
 
     // https://dev.w3.org/csswg/cssom/#dom-cssstyledeclaration-removeproperty
     fn RemoveProperty(&self, property: DOMString) -> Fallible<DOMString> {
         // Step 1
         if self.readonly {
             return Err(Error::NoModificationAllowed);
         }
 
-        let id = if let Ok(id) = PropertyId::parse(property.into()) {
+        let id = if let Ok(id) = PropertyId::parse(&property) {
             id
         } else {
             // Unkwown property, cannot be there to remove.
             return Ok(DOMString::new())
         };
 
         let mut string = String::new();
         self.owner.mutate_associated_block(|mut pdb, mut changed| {
--- a/servo/components/script/dom/dedicatedworkerglobalscope.rs
+++ b/servo/components/script/dom/dedicatedworkerglobalscope.rs
@@ -155,16 +155,17 @@ impl DedicatedWorkerGlobalScope {
                             parent_sender: Box<ScriptChan + Send>,
                             own_sender: Sender<(TrustedWorkerAddress, WorkerScriptMsg)>,
                             receiver: Receiver<(TrustedWorkerAddress, WorkerScriptMsg)>,
                             worker_load_origin: WorkerScriptLoadOrigin,
                             closing: Arc<AtomicBool>) {
         let serialized_worker_url = worker_url.to_string();
         let name = format!("WebWorker for {}", serialized_worker_url);
         let top_level_browsing_context_id = TopLevelBrowsingContextId::installed();
+        let origin = GlobalScope::current().expect("No current global object").origin().immutable().clone();
 
         thread::Builder::new().name(name).spawn(move || {
             thread_state::initialize(thread_state::SCRIPT | thread_state::IN_WORKER);
 
             if let Some(top_level_browsing_context_id) = top_level_browsing_context_id {
                 TopLevelBrowsingContextId::install(top_level_browsing_context_id);
             }
 
@@ -174,20 +175,20 @@ impl DedicatedWorkerGlobalScope {
             let WorkerScriptLoadOrigin { referrer_url, referrer_policy, pipeline_id } = worker_load_origin;
 
             let request = RequestInit {
                 url: worker_url.clone(),
                 type_: RequestType::Script,
                 destination: Destination::Worker,
                 credentials_mode: CredentialsMode::Include,
                 use_url_credentials: true,
-                origin: worker_url,
                 pipeline_id: pipeline_id,
                 referrer_url: referrer_url,
                 referrer_policy: referrer_policy,
+                origin,
                 .. RequestInit::default()
             };
 
             let (metadata, bytes) = match load_whole_resource(request,
                                                               &init.resource_threads.sender()) {
                 Err(_) => {
                     println!("error loading script {}", serialized_worker_url);
                     parent_sender.send(CommonScriptMsg::RunnableMsg(WorkerEvent,
--- a/servo/components/script/dom/element.rs
+++ b/servo/components/script/dom/element.rs
@@ -108,17 +108,17 @@ use style::properties::{Importance, Prop
 use style::properties::longhands::{self, background_image, border_spacing, font_family, font_size, overflow_x};
 use style::rule_tree::CascadeLevel;
 use style::selector_parser::{NonTSPseudoClass, PseudoElement, RestyleDamage, SelectorImpl, SelectorParser};
 use style::selector_parser::extended_filtering;
 use style::shared_lock::{SharedRwLock, Locked};
 use style::stylearc::Arc;
 use style::thread_state;
 use style::values::{CSSFloat, Either};
-use style::values::specified;
+use style::values::{specified, computed};
 use stylesheet_loader::StylesheetOwner;
 
 // TODO: Update focus state when the top-level browsing context gains or loses system focus,
 // and when the element enters or leaves a browsing context container.
 // https://html.spec.whatwg.org/multipage/#selector-focus
 
 #[dom_struct]
 pub struct Element {
@@ -571,17 +571,17 @@ impl LayoutElementHelpers for LayoutJS<E
             LengthOrPercentageOrAuto::Auto
         };
 
         // FIXME(emilio): Use from_computed value here and below.
         match width {
             LengthOrPercentageOrAuto::Auto => {}
             LengthOrPercentageOrAuto::Percentage(percentage) => {
                 let width_value =
-                    specified::LengthOrPercentageOrAuto::Percentage(specified::Percentage(percentage));
+                    specified::LengthOrPercentageOrAuto::Percentage(computed::Percentage(percentage));
                 hints.push(from_declaration(
                     shared_lock,
                     PropertyDeclaration::Width(width_value)));
             }
             LengthOrPercentageOrAuto::Length(length) => {
                 let width_value = specified::LengthOrPercentageOrAuto::Length(
                     specified::NoCalcLength::Absolute(specified::AbsoluteLength::Px(length.to_f32_px())));
                 hints.push(from_declaration(
@@ -600,17 +600,17 @@ impl LayoutElementHelpers for LayoutJS<E
         } else {
             LengthOrPercentageOrAuto::Auto
         };
 
         match height {
             LengthOrPercentageOrAuto::Auto => {}
             LengthOrPercentageOrAuto::Percentage(percentage) => {
                 let height_value =
-                    specified::LengthOrPercentageOrAuto::Percentage(specified::Percentage(percentage));
+                    specified::LengthOrPercentageOrAuto::Percentage(computed::Percentage(percentage));
                 hints.push(from_declaration(
                     shared_lock,
                     PropertyDeclaration::Height(height_value)));
             }
             LengthOrPercentageOrAuto::Length(length) => {
                 let height_value = specified::LengthOrPercentageOrAuto::Length(
                     specified::NoCalcLength::Absolute(specified::AbsoluteLength::Px(length.to_f32_px())));
                 hints.push(from_declaration(
--- a/servo/components/script/dom/eventsource.rs
+++ b/servo/components/script/dom/eventsource.rs
@@ -382,17 +382,17 @@ impl EventSource {
             CorsSettings::UseCredentials
         } else {
             CorsSettings::Anonymous
         };
         // Step 8
         // TODO: Step 9 set request's client settings
         let mut request = RequestInit {
             url: url_record,
-            origin: global.get_url(),
+            origin: global.origin().immutable().clone(),
             pipeline_id: Some(global.pipeline_id()),
             // https://html.spec.whatwg.org/multipage/#create-a-potential-cors-request
             use_url_credentials: true,
             mode: RequestMode::CorsMode,
             credentials_mode: if cors_attribute_state == CorsSettings::Anonymous {
                 CredentialsMode::CredentialsSameOrigin
             } else {
                 CredentialsMode::Include
--- a/servo/components/script/dom/htmlimageelement.rs
+++ b/servo/components/script/dom/htmlimageelement.rs
@@ -254,17 +254,17 @@ impl HTMLImageElement {
             wrapper: Some(window.get_runnable_wrapper()),
         };
         ROUTER.add_route(action_receiver.to_opaque(), box move |message| {
             listener.notify_fetch(message.to().unwrap());
         });
 
         let request = RequestInit {
             url: img_url.clone(),
-            origin: document.url().clone(),
+            origin: document.origin().immutable().clone(),
             type_: RequestType::Image,
             pipeline_id: Some(document.global().pipeline_id()),
             .. RequestInit::default()
         };
 
         // This is a background load because the load blocker already fulfills the
         // purpose of delaying the document's load event.
         document.loader().fetch_async_background(request, action_sender);
--- a/servo/components/script/dom/htmlmediaelement.rs
+++ b/servo/components/script/dom/htmlmediaelement.rs
@@ -545,17 +545,17 @@ impl HTMLMediaElement {
             };
 
             let request = RequestInit {
                 url: url.clone(),
                 type_: ty,
                 destination: Destination::Media,
                 credentials_mode: CredentialsMode::Include,
                 use_url_credentials: true,
-                origin: document.url(),
+                origin: document.origin().immutable().clone(),
                 pipeline_id: Some(self.global().pipeline_id()),
                 referrer_url: Some(document.url()),
                 referrer_policy: document.get_referrer_policy(),
                 .. RequestInit::default()
             };
 
             document.fetch_async(LoadType::Media(url), request, action_sender);
         } else {
--- a/servo/components/script/dom/htmlscriptelement.rs
+++ b/servo/components/script/dom/htmlscriptelement.rs
@@ -248,17 +248,17 @@ fn fetch_a_classic_script(script: &HTMLS
             None => RequestMode::NoCors,
         },
         // https://html.spec.whatwg.org/multipage/#create-a-potential-cors-request
         // Step 3-4
         credentials_mode: match cors_setting {
             Some(CorsSettings::Anonymous) => CredentialsMode::CredentialsSameOrigin,
             _ => CredentialsMode::Include,
         },
-        origin: doc.url(),
+        origin: doc.origin().immutable().clone(),
         pipeline_id: Some(script.global().pipeline_id()),
         referrer_url: Some(doc.url()),
         referrer_policy: doc.get_referrer_policy(),
         integrity_metadata: integrity_metadata,
         .. RequestInit::default()
     };
 
     // TODO: Step 3, Add custom steps to perform fetch
--- a/servo/components/script/dom/serviceworkerglobalscope.rs
+++ b/servo/components/script/dom/serviceworkerglobalscope.rs
@@ -146,33 +146,34 @@ impl ServiceWorkerGlobalScope {
                             swmanager_sender: IpcSender<ServiceWorkerMsg>,
                             scope_url: ServoUrl) {
         let ScopeThings { script_url,
                           init,
                           worker_load_origin,
                           .. } = scope_things;
 
         let serialized_worker_url = script_url.to_string();
+        let origin = GlobalScope::current().expect("No current global object").origin().immutable().clone();
         thread::Builder::new().name(format!("ServiceWorker for {}", serialized_worker_url)).spawn(move || {
             thread_state::initialize(SCRIPT | IN_WORKER);
             let roots = RootCollection::new();
             let _stack_roots_tls = StackRootTLS::new(&roots);
 
             let WorkerScriptLoadOrigin { referrer_url, referrer_policy, pipeline_id } = worker_load_origin;
 
             let request = RequestInit {
                 url: script_url.clone(),
                 type_: RequestType::Script,
                 destination: Destination::ServiceWorker,
                 credentials_mode: CredentialsMode::Include,
                 use_url_credentials: true,
-                origin: script_url,
                 pipeline_id: pipeline_id,
                 referrer_url: referrer_url,
                 referrer_policy: referrer_policy,
+                origin,
                 .. RequestInit::default()
             };
 
             let (url, source) = match load_whole_resource(request,
                                                           &init.resource_threads.sender()) {
                 Err(_) => {
                     println!("error loading script {}", serialized_worker_url);
                     return;
--- a/servo/components/script/dom/workerglobalscope.rs
+++ b/servo/components/script/dom/workerglobalscope.rs
@@ -206,17 +206,17 @@ impl WorkerGlobalScopeMethods for Worker
         for url in urls {
             let global_scope = self.upcast::<GlobalScope>();
             let request = NetRequestInit {
                 url: url.clone(),
                 type_: RequestType::Script,
                 destination: Destination::Script,
                 credentials_mode: CredentialsMode::Include,
                 use_url_credentials: true,
-                origin: self.worker_url.clone(),
+                origin: global_scope.origin().immutable().clone(),
                 pipeline_id: Some(self.upcast::<GlobalScope>().pipeline_id()),
                 referrer_url: None,
                 referrer_policy: None,
                 .. NetRequestInit::default()
             };
             let (url, source) = match load_whole_resource(request,
                                                           &global_scope.resource_threads().sender()) {
                 Err(_) => return Err(Error::Network),
--- a/servo/components/script/dom/worklet.rs
+++ b/servo/components/script/dom/worklet.rs
@@ -569,27 +569,24 @@ impl WorkletThread {
         debug!("Fetching from {}.", script_url);
         // Step 1.
         // TODO: Settings object?
 
         // Step 2.
         // TODO: Fetch a module graph, not just a single script.
         // TODO: Fetch the script asynchronously?
         // TODO: Caching.
-        // TODO: Avoid re-parsing the origin as a URL.
         let resource_fetcher = self.global_init.resource_threads.sender();
-        let origin_url = ServoUrl::parse(&*origin.unicode_serialization())
-            .unwrap_or_else(|_| ServoUrl::parse("about:blank").unwrap());
         let request = RequestInit {
             url: script_url,
             type_: RequestType::Script,
             destination: Destination::Script,
             mode: RequestMode::CorsMode,
-            origin: origin_url,
             credentials_mode: credentials.into(),
+            origin,
             .. RequestInit::default()
         };
         let script = load_whole_resource(request, &resource_fetcher).ok()
             .and_then(|(_, bytes)| String::from_utf8(bytes).ok());
 
         // Step 4.
         // NOTE: the spec parses and executes the script in separate steps,
         // but our JS API doesn't separate these, so we do the steps out of order.
--- a/servo/components/script/dom/xmlhttprequest.rs
+++ b/servo/components/script/dom/xmlhttprequest.rs
@@ -589,17 +589,17 @@ impl XMLHttpRequestMethods for XMLHttpRe
             // XXXManishearth actually "subresource", but it doesn't exist
             // https://github.com/whatwg/xhr/issues/71
             destination: Destination::None,
             synchronous: self.sync.get(),
             mode: RequestMode::CorsMode,
             use_cors_preflight: has_handlers,
             credentials_mode: credentials_mode,
             use_url_credentials: use_url_credentials,
-            origin: self.global().get_url(),
+            origin: self.global().origin().immutable().clone(),
             referrer_url: self.referrer_url.clone(),
             referrer_policy: self.referrer_policy.clone(),
             pipeline_id: Some(self.global().pipeline_id()),
             .. RequestInit::default()
         };
 
         if bypass_cross_origin_check {
             request.mode = RequestMode::Navigate;
--- a/servo/components/script/fetch.rs
+++ b/servo/components/script/fetch.rs
@@ -51,20 +51,17 @@ fn request_init_from_request(request: Ne
         body: request.body.clone(),
         type_: request.type_,
         destination: request.destination,
         synchronous: request.synchronous,
         mode: request.mode,
         use_cors_preflight: request.use_cors_preflight,
         credentials_mode: request.credentials_mode,
         use_url_credentials: request.use_url_credentials,
-        // TODO: NetTraitsRequestInit and NetTraitsRequest have different "origin"
-        // ... NetTraitsRequestInit.origin: Url
-        // ... NetTraitsRequest.origin: RefCell<Origin>
-        origin: request.url(),
+        origin: GlobalScope::current().expect("No current global object").origin().immutable().clone(),
         referrer_url: from_referrer_to_referrer_url(&request),
         referrer_policy: request.referrer_policy,
         pipeline_id: request.pipeline_id,
         redirect_mode: request.redirect_mode,
         ..NetTraitsRequestInit::default()
     }
 }
 
--- a/servo/components/script/layout_image.rs
+++ b/servo/components/script/layout_image.rs
@@ -65,17 +65,17 @@ pub fn fetch_image_for_layout(url: Servo
         wrapper: Some(window.get_runnable_wrapper()),
     };
     ROUTER.add_route(action_receiver.to_opaque(), box move |message| {
         listener.notify_fetch(message.to().unwrap());
     });
 
     let request = FetchRequestInit {
         url: url,
-        origin: document.url().clone(),
+        origin: document.origin().immutable().clone(),
         type_: RequestType::Image,
         pipeline_id: Some(document.global().pipeline_id()),
         .. FetchRequestInit::default()
     };
 
     // Layout image loads do not delay the document load event.
     document.mut_loader().fetch_async_background(request, action_sender);
 }
--- a/servo/components/script/script_thread.rs
+++ b/servo/components/script/script_thread.rs
@@ -1923,16 +1923,22 @@ impl ScriptThread {
         let DOMManipulationTaskSource(ref dom_sender) = self.dom_manipulation_task_source;
         let UserInteractionTaskSource(ref user_sender) = self.user_interaction_task_source;
         let HistoryTraversalTaskSource(ref history_sender) = self.history_traversal_task_source;
 
         let (ipc_timer_event_chan, ipc_timer_event_port) = ipc::channel().unwrap();
         ROUTER.route_ipc_receiver_to_mpsc_sender(ipc_timer_event_port,
                                                  self.timer_event_chan.clone());
 
+        let origin = if final_url.as_str() == "about:blank" {
+            incomplete.origin.clone()
+        } else {
+            MutableOrigin::new(final_url.origin())
+        };
+
         // Create the window and document objects.
         let window = Window::new(self.js_runtime.clone(),
                                  MainThreadScriptChan(sender.clone()),
                                  DOMManipulationTaskSource(dom_sender.clone()),
                                  UserInteractionTaskSource(user_sender.clone()),
                                  self.networking_task_source.clone(),
                                  HistoryTraversalTaskSource(history_sender.clone()),
                                  self.file_reading_task_source.clone(),
@@ -1946,17 +1952,17 @@ impl ScriptThread {
                                  self.constellation_chan.clone(),
                                  self.control_chan.clone(),
                                  self.scheduler_chan.clone(),
                                  ipc_timer_event_chan,
                                  incomplete.layout_chan,
                                  incomplete.pipeline_id,
                                  incomplete.parent_info,
                                  incomplete.window_size,
-                                 incomplete.origin.clone(),
+                                 origin,
                                  self.webvr_thread.clone());
 
         // Initialize the browsing context for the window.
         let window_proxy = self.local_window_proxy(&window,
                                                    incomplete.browsing_context_id,
                                                    incomplete.top_level_browsing_context_id,
                                                    incomplete.parent_info);
         window.init_window_proxy(&window_proxy);
@@ -2278,23 +2284,23 @@ impl ScriptThread {
     fn pre_page_load(&self, incomplete: InProgressLoad, load_data: LoadData) {
         let id = incomplete.pipeline_id.clone();
         let mut req_init = RequestInit {
             url: load_data.url.clone(),
             method: load_data.method,
             destination: Destination::Document,
             credentials_mode: CredentialsMode::Include,
             use_url_credentials: true,
-            origin: load_data.url.clone(),
             pipeline_id: Some(id),
             referrer_url: load_data.referrer_url,
             referrer_policy: load_data.referrer_policy,
             headers: load_data.headers,
             body: load_data.data,
             redirect_mode: RedirectMode::Manual,
+            origin: incomplete.origin.immutable().clone(),
             .. RequestInit::default()
         };
 
         if req_init.url.scheme() == "javascript" {
             req_init.url = ServoUrl::parse("about:blank").unwrap();
         }
 
         let context = ParserContext::new(id, load_data.url);
--- a/servo/components/script/stylesheet_loader.rs
+++ b/servo/components/script/stylesheet_loader.rs
@@ -256,17 +256,17 @@ impl<'a> StylesheetLoader<'a> {
                 None => RequestMode::NoCors,
             },
             // https://html.spec.whatwg.org/multipage/#create-a-potential-cors-request
             // Step 3-4
             credentials_mode: match cors_setting {
                 Some(CorsSettings::Anonymous) => CredentialsMode::CredentialsSameOrigin,
                 _ => CredentialsMode::Include,
             },
-            origin: document.url(),
+            origin: document.origin().immutable().clone(),
             pipeline_id: Some(self.elem.global().pipeline_id()),
             referrer_url: Some(document.url()),
             referrer_policy: referrer_policy,
             integrity_metadata: integrity_metadata,
             .. RequestInit::default()
         };
 
         document.fetch_async(LoadType::Stylesheet(url), request, action_sender);
--- a/servo/components/style/gecko/conversions.rs
+++ b/servo/components/style/gecko/conversions.rs
@@ -13,21 +13,20 @@ use gecko::values::{convert_rgba_to_nsco
 use gecko_bindings::bindings::{Gecko_CreateGradient, Gecko_SetGradientImageValue, Gecko_SetLayerImageImageValue};
 use gecko_bindings::bindings::{Gecko_InitializeImageCropRect, Gecko_SetImageElement};
 use gecko_bindings::structs::{nsCSSUnit, nsStyleCoord_CalcValue, nsStyleImage};
 use gecko_bindings::structs::{nsresult, SheetType};
 use gecko_bindings::sugar::ns_style_coord::{CoordDataValue, CoordData, CoordDataMut};
 use std::f32::consts::PI;
 use stylesheets::{Origin, RulesMutateError};
 use values::computed::{Angle, CalcLengthOrPercentage, Gradient, Image};
-use values::computed::{LengthOrPercentage, LengthOrPercentageOrAuto};
+use values::computed::{LengthOrPercentage, LengthOrPercentageOrAuto, Percentage};
 use values::generics::grid::TrackSize;
 use values::generics::image::{CompatMode, Image as GenericImage, GradientItem};
 use values::generics::rect::Rect;
-use values::specified::length::Percentage;
 use values::specified::url::SpecifiedUrl;
 
 impl From<CalcLengthOrPercentage> for nsStyleCoord_CalcValue {
     fn from(other: CalcLengthOrPercentage) -> nsStyleCoord_CalcValue {
         let has_percentage = other.percentage.is_some();
         nsStyleCoord_CalcValue {
             mLength: other.unclamped_length().0,
             mPercent: other.percentage.map_or(0., |p| p.0),
--- a/servo/components/style/gecko/values.rs
+++ b/servo/components/style/gecko/values.rs
@@ -13,23 +13,22 @@ use gecko_bindings::structs::{CounterSty
 use gecko_bindings::structs::{StyleGridTrackBreadth, StyleShapeRadius};
 use gecko_bindings::sugar::ns_style_coord::{CoordData, CoordDataMut, CoordDataValue};
 use media_queries::Device;
 use nsstring::{nsACString, nsCString};
 use std::cmp::max;
 use values::{Auto, Either, ExtremumLength, None_, Normal};
 use values::computed::{Angle, LengthOrPercentage, LengthOrPercentageOrAuto};
 use values::computed::{LengthOrPercentageOrNone, Number, NumberOrPercentage};
-use values::computed::{MaxLength, MozLength};
+use values::computed::{MaxLength, MozLength, Percentage};
 use values::computed::basic_shape::ShapeRadius as ComputedShapeRadius;
 use values::generics::CounterStyleOrNone;
 use values::generics::basic_shape::ShapeRadius;
 use values::generics::gecko::ScrollSnapPoint;
 use values::generics::grid::{TrackBreadth, TrackKeyword};
-use values::specified::Percentage;
 
 /// A trait that defines an interface to convert from and to `nsStyleCoord`s.
 pub trait GeckoStyleCoordConvertible : Sized {
     /// Convert this to a `nsStyleCoord`.
     fn to_gecko_style_coord<T: CoordDataMut>(&self, coord: &mut T);
     /// Given a `nsStyleCoord`, try to get a value of this type..
     fn from_gecko_style_coord<T: CoordData>(coord: &T) -> Option<Self>;
 }
--- a/servo/components/style/gecko_bindings/sugar/ns_css_value.rs
+++ b/servo/components/style/gecko_bindings/sugar/ns_css_value.rs
@@ -9,18 +9,17 @@ use gecko_bindings::bindings;
 use gecko_bindings::structs;
 use gecko_bindings::structs::{nsCSSValue, nsCSSUnit};
 use gecko_bindings::structs::{nsCSSValue_Array, nsCSSValueList, nscolor};
 use gecko_string_cache::Atom;
 use std::marker::PhantomData;
 use std::mem;
 use std::ops::{Index, IndexMut};
 use std::slice;
-use values::computed::{Angle, LengthOrPercentage};
-use values::specified::length::Percentage;
+use values::computed::{Angle, LengthOrPercentage, Percentage};
 use values::specified::url::SpecifiedUrl;
 
 impl nsCSSValue {
     /// Create a CSSValue with null unit, useful to be used as a return value.
     #[inline]
     pub fn null() -> Self {
         unsafe { mem::zeroed() }
     }
--- a/servo/components/style/properties/declaration_block.rs
+++ b/servo/components/style/properties/declaration_block.rs
@@ -3,17 +3,17 @@
  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 
 //! A property declaration block.
 
 #![deny(missing_docs)]
 
 use context::QuirksMode;
 use cssparser::{DeclarationListParser, parse_important, ParserInput, CompactCowStr};
-use cssparser::{Parser, AtRuleParser, DeclarationParser, Delimiter};
+use cssparser::{Parser, AtRuleParser, DeclarationParser, Delimiter, ParseError as CssParseError};
 use error_reporting::{ParseErrorReporter, ContextualParseError};
 use parser::{ParserContext, log_css_error};
 use properties::animated_properties::AnimationValue;
 use selectors::parser::SelectorParseError;
 use shared_lock::Locked;
 use smallvec::SmallVec;
 use std::fmt;
 use std::slice::Iter;
@@ -930,23 +930,38 @@ struct PropertyDeclarationParser<'a, 'b:
 
 /// Default methods reject all at rules.
 impl<'a, 'b, 'i> AtRuleParser<'i> for PropertyDeclarationParser<'a, 'b> {
     type Prelude = ();
     type AtRule = Importance;
     type Error = SelectorParseError<'i, StyleParseError<'i>>;
 }
 
+/// Based on NonMozillaVendorIdentifier from Gecko's CSS parser.
+fn is_non_mozilla_vendor_identifier(name: &str) -> bool {
+    (name.starts_with("-") && !name.starts_with("-moz-")) ||
+        name.starts_with("_")
+}
+
 impl<'a, 'b, 'i> DeclarationParser<'i> for PropertyDeclarationParser<'a, 'b> {
     type Declaration = Importance;
     type Error = SelectorParseError<'i, StyleParseError<'i>>;
 
     fn parse_value<'t>(&mut self, name: CompactCowStr<'i>, input: &mut Parser<'i, 't>)
                        -> Result<Importance, ParseError<'i>> {
-        let id = PropertyId::parse(name)?;
+        let id = match PropertyId::parse(&name) {
+            Ok(id) => id,
+            Err(()) => {
+                return Err(if is_non_mozilla_vendor_identifier(&name) {
+                    PropertyDeclarationParseError::UnknownVendorProperty
+                } else {
+                    PropertyDeclarationParseError::UnknownProperty(name)
+                }.into());
+            }
+        };
         input.parse_until_before(Delimiter::Bang, |input| {
             PropertyDeclaration::parse_into(self.declarations, id, self.context, input)
                 .map_err(|e| e.into())
         })?;
         let importance = match input.try(parse_important) {
             Ok(()) => Importance::Important,
             Err(_) => Importance::Normal,
         };
@@ -971,16 +986,25 @@ pub fn parse_property_declaration_list(c
     let mut iter = DeclarationListParser::new(input, parser);
     while let Some(declaration) = iter.next() {
         match declaration {
             Ok(importance) => {
                 block.extend(iter.parser.declarations.drain(), importance);
             }
             Err(err) => {
                 iter.parser.declarations.clear();
+
+                // If the unrecognized property looks like a vendor-specific property,
+                // silently ignore it instead of polluting the error output.
+                if let CssParseError::Custom(SelectorParseError::Custom(
+                    StyleParseError::PropertyDeclaration(
+                        PropertyDeclarationParseError::UnknownVendorProperty))) = err.error {
+                    continue;
+                }
+
                 let pos = err.span.start;
                 let error = ContextualParseError::UnsupportedPropertyDeclaration(
                     iter.input.slice(err.span), err.error);
                 log_css_error(iter.input, pos, error, &context);
             }
         }
     }
     block
--- a/servo/components/style/properties/gecko.mako.rs
+++ b/servo/components/style/properties/gecko.mako.rs
@@ -61,17 +61,17 @@ use rule_tree::StrongRuleNode;
 use std::fmt::{self, Debug};
 use std::mem::{forget, transmute, zeroed};
 use std::ptr;
 use stylearc::Arc;
 use std::cmp;
 use values::{Auto, CustomIdent, Either, KeyframesName};
 use values::computed::ToComputedValue;
 use values::computed::effects::{BoxShadow, Filter, SimpleShadow};
-use values::specified::length::Percentage;
+use values::computed::length::Percentage;
 use computed_values::border_style;
 
 pub mod style_structs {
     % for style_struct in data.style_structs:
     pub use super::${style_struct.gecko_struct_name} as ${style_struct.name};
     % endfor
 }
 
@@ -2866,17 +2866,17 @@ fn static_assert() {
                     } else if feature.0 == atom!("transform") {
                         self.gecko.mWillChangeBitField |= NS_STYLE_WILL_CHANGE_TRANSFORM as u8;
                     }
 
                     unsafe {
                         Gecko_AppendWillChange(&mut self.gecko, feature.0.as_ptr());
                     }
 
-                    if let Ok(prop_id) = PropertyId::parse(feature.0.to_string().into()) {
+                    if let Ok(prop_id) = PropertyId::parse(&feature.0.to_string()) {
                         match prop_id.as_shorthand() {
                             Ok(shorthand) => {
                                 for longhand in shorthand.longhands() {
                                     self.gecko.mWillChangeBitField |=
                                         will_change_bitfield_from_prop_flags(longhand);
                                 }
                             },
                             Err(longhand_or_custom) => {
--- a/servo/components/style/properties/helpers/animated_properties.mako.rs
+++ b/servo/components/style/properties/helpers/animated_properties.mako.rs
@@ -36,22 +36,21 @@ use values::{CSSFloat, CustomIdent, Eith
 use values::animated::ToAnimatedValue;
 use values::animated::effects::BoxShadowList as AnimatedBoxShadowList;
 use values::animated::effects::Filter as AnimatedFilter;
 use values::animated::effects::FilterList as AnimatedFilterList;
 use values::animated::effects::TextShadowList as AnimatedTextShadowList;
 use values::computed::{Angle, LengthOrPercentageOrAuto, LengthOrPercentageOrNone};
 use values::computed::{BorderCornerRadius, ClipRect};
 use values::computed::{CalcLengthOrPercentage, Color, Context, ComputedValueAsSpecified};
-use values::computed::{LengthOrPercentage, MaxLength, MozLength, ToComputedValue};
+use values::computed::{LengthOrPercentage, MaxLength, MozLength, Percentage, ToComputedValue};
 use values::generics::{SVGPaint, SVGPaintKind};
 use values::generics::border::BorderCornerRadius as GenericBorderCornerRadius;
 use values::generics::effects::Filter;
 use values::generics::position as generic_position;
-use values::specified::length::Percentage;
 
 
 /// A longhand property whose animation type is not "none".
 ///
 /// NOTE: This includes the 'display' property since it is animatable from SMIL even though it is
 /// not animatable from CSS animations or Web Animations. CSS transitions also does not allow
 /// animating 'display', but for CSS transitions we have the separate TransitionProperty type.
 #[derive(Clone, Debug, PartialEq, Eq, Hash)]
@@ -1681,17 +1680,16 @@ fn add_weighted_transform_lists(from_lis
                 }
                 _ => {
                     // This should be unreachable due to the can_interpolate_list() call.
                     unreachable!();
                 }
             }
         }
     } else {
-        use values::specified::Percentage;
         let from_transform_list = TransformList(Some(from_list.to_vec()));
         let to_transform_list = TransformList(Some(to_list.to_vec()));
         result.push(
             TransformOperation::InterpolateMatrix { from_list: from_transform_list,
                                                     to_list: to_transform_list,
                                                     progress: Percentage(other_portion as f32) });
     }
 
--- a/servo/components/style/properties/longhand/border.mako.rs
+++ b/servo/components/style/properties/longhand/border.mako.rs
@@ -273,17 +273,17 @@
     initial_value="computed::BorderImageSideWidth::one().into()",
     initial_specified_value="specified::BorderImageSideWidth::one().into()",
     spec="https://drafts.csswg.org/css-backgrounds/#border-image-width",
     animation_value_type="discrete",
     boxed=True)}
 
 ${helpers.predefined_type("border-image-slice", "BorderImageSlice",
     initial_value="computed::NumberOrPercentage::Percentage(computed::Percentage(1.)).into()",
-    initial_specified_value="specified::NumberOrPercentage::Percentage(specified::Percentage(1.)).into()",
+    initial_specified_value="specified::NumberOrPercentage::Percentage(specified::Percentage::new(1.)).into()",
     spec="https://drafts.csswg.org/css-backgrounds/#border-image-slice",
     animation_value_type="discrete",
     boxed=True)}
 
 #[cfg(feature = "gecko")]
 impl ::values::computed::BorderImageWidth {
     pub fn to_gecko_rect(&self, sides: &mut ::gecko_bindings::structs::nsStyleSides) {
         use gecko_bindings::sugar::ns_style_coord::{CoordDataMut, CoordDataValue};
--- a/servo/components/style/properties/longhand/box.mako.rs
+++ b/servo/components/style/properties/longhand/box.mako.rs
@@ -731,28 +731,28 @@
 <%helpers:longhand name="transform" extra_prefixes="webkit"
                    animation_value_type="ComputedValue"
                    flags="CREATES_STACKING_CONTEXT FIXPOS_CB"
                    spec="https://drafts.csswg.org/css-transforms/#propdef-transform">
     use app_units::Au;
     use values::computed::{LengthOrPercentageOrNumber as ComputedLoPoNumber, LengthOrNumber as ComputedLoN};
     use values::computed::{LengthOrPercentage as ComputedLoP, Length as ComputedLength};
     use values::generics::transform::Matrix;
-    use values::specified::{Angle, Integer, Length, LengthOrPercentage, Percentage};
+    use values::specified::{Angle, Integer, Length, LengthOrPercentage};
     use values::specified::{LengthOrNumber, LengthOrPercentageOrNumber as LoPoNumber, Number};
     use style_traits::ToCss;
     use style_traits::values::Css;
 
     use std::fmt;
 
     pub mod computed_value {
         use app_units::Au;
         use values::CSSFloat;
         use values::computed;
-        use values::computed::{Length, LengthOrPercentage, Percentage};
+        use values::computed::{Length, LengthOrPercentage};
 
         #[derive(Clone, Copy, Debug, PartialEq)]
         #[cfg_attr(feature = "servo", derive(HeapSizeOf))]
         pub struct ComputedMatrix {
             pub m11: CSSFloat, pub m12: CSSFloat, pub m13: CSSFloat, pub m14: CSSFloat,
             pub m21: CSSFloat, pub m22: CSSFloat, pub m23: CSSFloat, pub m24: CSSFloat,
             pub m31: CSSFloat, pub m32: CSSFloat, pub m33: CSSFloat, pub m34: CSSFloat,
             pub m41: CSSFloat, pub m42: CSSFloat, pub m43: CSSFloat, pub m44: CSSFloat,
@@ -812,17 +812,17 @@
             //                                    Scale(...) ],
             //                       to_list: [ AccumulateMatrix { from_list: ...,
             //                                                     to_list: [ InterpolateMatrix,
             //                                                                 ... ],
             //                                                     count: ... } ],
             //                       progress: ... } ]
             InterpolateMatrix { from_list: T,
                                 to_list: T,
-                                progress: Percentage },
+                                progress: computed::Percentage },
             // For accumulate operation of mismatched transform lists.
             AccumulateMatrix { from_list: T,
                                to_list: T,
                                count: computed::Integer },
         }
 
         #[derive(Clone, Debug, PartialEq)]
         #[cfg_attr(feature = "servo", derive(HeapSizeOf))]
@@ -899,17 +899,17 @@
         /// Part of CSS Transform Module Level 2 and defined at
         /// [§ 13.1. 3D Transform Function](https://drafts.csswg.org/css-transforms-2/#funcdef-perspective).
         ///
         /// The value must be greater than or equal to zero.
         Perspective(specified::Length),
         /// A intermediate type for interpolation of mismatched transform lists.
         InterpolateMatrix { from_list: SpecifiedValue,
                             to_list: SpecifiedValue,
-                            progress: Percentage },
+                            progress: computed::Percentage },
         /// A intermediate type for accumulation of mismatched transform lists.
         AccumulateMatrix { from_list: SpecifiedValue,
                            to_list: SpecifiedValue,
                            count: Integer },
     }
 
     impl ToCss for computed_value::T {
         fn to_css<W>(&self, _: &mut W) -> fmt::Result where W: fmt::Write {
--- a/servo/components/style/properties/longhand/font.mako.rs
+++ b/servo/components/style/properties/longhand/font.mako.rs
@@ -561,18 +561,17 @@ macro_rules! impl_gecko_keyword_conversi
 <%helpers:longhand name="font-size" need_clone="True" animation_value_type="ComputedValue"
                    allow_quirks="True" spec="https://drafts.csswg.org/css-fonts/#propdef-font-size">
     use app_units::Au;
     use properties::longhands::system_font::SystemFont;
     use properties::style_structs::Font;
     use std::fmt;
     use style_traits::{HasViewportPercentage, ToCss};
     use values::FONT_MEDIUM_PX;
-    use values::specified::{AllowQuirks, FontRelativeLength, LengthOrPercentage};
-    use values::specified::{NoCalcLength, Percentage};
+    use values::specified::{AllowQuirks, FontRelativeLength, LengthOrPercentage, NoCalcLength};
     use values::specified::length::FontBaseSize;
 
     impl ToCss for SpecifiedValue {
         fn to_css<W>(&self, dest: &mut W) -> fmt::Result where W: fmt::Write {
             match *self {
                 SpecifiedValue::Length(ref lop) => lop.to_css(dest),
                 SpecifiedValue::Keyword(kw, _) => kw.to_css(dest),
                 SpecifiedValue::Smaller => dest.write_str("smaller"),
@@ -805,18 +804,18 @@ macro_rules! impl_gecko_keyword_conversi
                 }
                 SpecifiedValue::Length(LengthOrPercentage::Length(
                         NoCalcLength::ServoCharacterWidth(value))) => {
                     value.to_computed_value(base_size.resolve(context))
                 }
                 SpecifiedValue::Length(LengthOrPercentage::Length(ref l)) => {
                     l.to_computed_value(context)
                 }
-                SpecifiedValue::Length(LengthOrPercentage::Percentage(Percentage(value))) => {
-                    base_size.resolve(context).scale_by(value)
+                SpecifiedValue::Length(LengthOrPercentage::Percentage(pc)) => {
+                    base_size.resolve(context).scale_by(pc.0)
                 }
                 SpecifiedValue::Length(LengthOrPercentage::Calc(ref calc)) => {
                     let calc = calc.to_computed_value(context);
                     calc.to_used_value(Some(base_size.resolve(context))).unwrap()
                 }
                 SpecifiedValue::Keyword(ref key, fraction) => {
                     key.to_computed_value(context).scale_by(fraction)
                 }
--- a/servo/components/style/properties/properties.mako.rs
+++ b/servo/components/style/properties/properties.mako.rs
@@ -15,17 +15,17 @@ use std::collections::HashSet;
 use std::fmt;
 use std::mem;
 use std::ops::Deref;
 use stylearc::{Arc, UniqueArc};
 
 use app_units::Au;
 #[cfg(feature = "servo")] use cssparser::RGBA;
 use cssparser::{Parser, TokenSerializationType, serialize_identifier};
-use cssparser::{ParserInput, CompactCowStr};
+use cssparser::ParserInput;
 #[cfg(feature = "servo")] use euclid::SideOffsets2D;
 use computed_values;
 use context::QuirksMode;
 use error_reporting::NullReporter;
 use font_metrics::FontMetricsProvider;
 #[cfg(feature = "gecko")] use gecko_bindings::bindings;
 #[cfg(feature = "gecko")] use gecko_bindings::structs::{self, nsCSSPropertyID};
 #[cfg(feature = "servo")] use logical_geometry::{LogicalMargin, PhysicalSide};
@@ -505,17 +505,17 @@ impl LonghandId {
     fn parse_value<'i, 't>(&self, context: &ParserContext, input: &mut Parser<'i, 't>)
                            -> Result<PropertyDeclaration, ParseError<'i>> {
         match *self {
             % for property in data.longhands:
                 LonghandId::${property.camel_case} => {
                     % if not property.derived_from:
                         longhands::${property.ident}::parse_declared(context, input)
                     % else:
-                        Err(PropertyDeclarationParseError::UnknownProperty.into())
+                        Err(PropertyDeclarationParseError::UnknownProperty("${property.ident}".into()).into())
                     % endif
                 }
             % endfor
         }
     }
 
     /// If this is a logical property, return the corresponding physical one in the given writing mode.
     /// Otherwise, return unchanged.
@@ -987,18 +987,18 @@ impl ToCss for PropertyId {
         }
     }
 }
 
 impl PropertyId {
     /// Returns a given property from the string `s`.
     ///
     /// Returns Err(()) for unknown non-custom properties
-    pub fn parse<'i>(property_name: CompactCowStr<'i>) -> Result<Self, ParseError<'i>> {
-        if let Ok(name) = ::custom_properties::parse_name(&property_name) {
+    pub fn parse(property_name: &str) -> Result<Self, ()> {
+        if let Ok(name) = ::custom_properties::parse_name(property_name) {
             return Ok(PropertyId::Custom(::custom_properties::Name::from(name)))
         }
 
         // FIXME(https://github.com/rust-lang/rust/issues/33156): remove this enum and use PropertyId
         // when stable Rust allows destructors in statics.
         pub enum StaticId {
             Longhand(LonghandId),
             Shorthand(ShorthandId),
@@ -1009,20 +1009,20 @@ impl PropertyId {
                     % for property in properties:
                         % for name in [property.name] + property.alias:
                             "${name}" => StaticId::${kind}(${kind}Id::${property.camel_case}),
                         % endfor
                     % endfor
                 % endfor
             }
         }
-        match static_id(&property_name) {
+        match static_id(property_name) {
             Some(&StaticId::Longhand(id)) => Ok(PropertyId::Longhand(id)),
             Some(&StaticId::Shorthand(id)) => Ok(PropertyId::Shorthand(id)),
-            None => Err(StyleParseError::UnknownProperty(property_name).into()),
+            None => Err(()),
         }
     }
 
     /// Returns a property id from Gecko's nsCSSPropertyID.
     #[cfg(feature = "gecko")]
     #[allow(non_upper_case_globals)]
     pub fn from_nscsspropertyid(id: nsCSSPropertyID) -> Result<Self, ()> {
         use gecko_bindings::structs::*;
@@ -1096,17 +1096,17 @@ impl PropertyId {
                 let mut s = String::new();
                 write!(&mut s, "--{}", name).unwrap();
                 s.into()
             }
         }
     }
 
     fn check_allowed_in(&self, rule_type: CssRuleType, stylesheet_origin: Origin)
-                        -> Result<(), PropertyDeclarationParseError> {
+                        -> Result<(), PropertyDeclarationParseError<'static>> {
         let id: NonCustomPropertyId;
         match *self {
             // Custom properties are allowed everywhere
             PropertyId::Custom(_) => return Ok(()),
 
             PropertyId::Shorthand(shorthand_id) => id = shorthand_id.into(),
             PropertyId::Longhand(longhand_id) => id = longhand_id.into(),
         }
@@ -1181,17 +1181,17 @@ impl PropertyId {
 
         if INTERNAL.contains(id) {
             if stylesheet_origin != Origin::UserAgent {
                 if EXPERIMENTAL.contains(id) {
                     if !passes_pref_check() {
                         return Err(PropertyDeclarationParseError::ExperimentalProperty);
                     }
                 } else {
-                    return Err(PropertyDeclarationParseError::UnknownProperty);
+                    return Err(PropertyDeclarationParseError::UnknownProperty(self.name().into()));
                 }
             }
         } else {
             if EXPERIMENTAL.contains(id) && !passes_pref_check() {
                 return Err(PropertyDeclarationParseError::ExperimentalProperty);
             }
         }
 
@@ -1457,19 +1457,19 @@ impl PropertyDeclaration {
     /// https://drafts.csswg.org/css-animations/#keyframes
     /// > The <declaration-list> inside of <keyframe-block> accepts any CSS property
     /// > except those defined in this specification,
     /// > but does accept the `animation-play-state` property and interprets it specially.
     ///
     /// This will not actually parse Importance values, and will always set things
     /// to Importance::Normal. Parsing Importance values is the job of PropertyDeclarationParser,
     /// we only set them here so that we don't have to reallocate
-    pub fn parse_into(declarations: &mut SourcePropertyDeclaration,
-                      id: PropertyId, context: &ParserContext, input: &mut Parser)
-                      -> Result<(), PropertyDeclarationParseError> {
+    pub fn parse_into<'i, 't>(declarations: &mut SourcePropertyDeclaration,
+                              id: PropertyId, context: &ParserContext, input: &mut Parser<'i, 't>)
+                              -> Result<(), PropertyDeclarationParseError<'i>> {
         assert!(declarations.is_empty());
         let rule_type = context.rule_type();
         debug_assert!(rule_type == CssRuleType::Keyframe ||
                       rule_type == CssRuleType::Page ||
                       rule_type == CssRuleType::Style,
                       "Declarations are only expected inside a keyframe, page, or style rule.");
         id.check_allowed_in(rule_type, context.stylesheet_origin)?;
         match id {
--- a/servo/components/style/stylesheets/keyframes_rule.rs
+++ b/servo/components/style/stylesheets/keyframes_rule.rs
@@ -12,16 +12,17 @@ use properties::{Importance, PropertyDec
 use properties::{PropertyDeclarationId, LonghandId, SourcePropertyDeclaration};
 use properties::LonghandIdSet;
 use properties::animated_properties::AnimatableLonghand;
 use properties::longhands::transition_timing_function::single_value::SpecifiedValue as SpecifiedTimingFunction;
 use selectors::parser::SelectorParseError;
 use shared_lock::{DeepCloneParams, DeepCloneWithLock, SharedRwLock, SharedRwLockReadGuard, Locked, ToCssWithGuard};
 use std::fmt;
 use style_traits::{PARSING_MODE_DEFAULT, ToCss, ParseError, StyleParseError};
+use style_traits::PropertyDeclarationParseError;
 use stylearc::Arc;
 use stylesheets::{CssRuleType, StylesheetContents};
 use stylesheets::rule_parser::VendorPrefix;
 use values::KeyframesName;
 
 /// A [`@keyframes`][keyframes] rule.
 ///
 /// [keyframes]: https://drafts.csswg.org/css-animations/#keyframes
@@ -527,17 +528,18 @@ impl<'a, 'b, 'i> AtRuleParser<'i> for Ke
 }
 
 impl<'a, 'b, 'i> DeclarationParser<'i> for KeyframeDeclarationParser<'a, 'b> {
     type Declaration = ();
     type Error = SelectorParseError<'i, StyleParseError<'i>>;
 
     fn parse_value<'t>(&mut self, name: CompactCowStr<'i>, input: &mut Parser<'i, 't>)
                        -> Result<(), ParseError<'i>> {
-        let id = PropertyId::parse(name.into())?;
+        let id = PropertyId::parse(&name)
+            .map_err(|()| PropertyDeclarationParseError::UnknownProperty(name))?;
         match PropertyDeclaration::parse_into(self.declarations, id, self.context, input) {
             Ok(()) => {
                 // In case there is still unparsed text in the declaration, we should roll back.
                 input.expect_exhausted().map_err(|e| e.into())
             }
             Err(_e) => Err(StyleParseError::UnspecifiedError.into())
         }
     }
--- a/servo/components/style/values/computed/length.rs
+++ b/servo/components/style/values/computed/length.rs
@@ -7,21 +7,62 @@
 use app_units::{Au, AU_PER_PX};
 use ordered_float::NotNaN;
 use std::fmt;
 use style_traits::ToCss;
 use style_traits::values::specified::AllowedLengthType;
 use super::{Number, ToComputedValue, Context};
 use values::{Auto, CSSFloat, Either, ExtremumLength, None_, Normal, specified};
 use values::specified::length::{AbsoluteLength, FontBaseSize, FontRelativeLength};
-use values::specified::length::{Percentage, ViewportPercentageLength};
+use values::specified::length::ViewportPercentageLength;
 
 pub use super::image::Image;
 pub use values::specified::{Angle, BorderStyle, Time, UrlOrNone};
 
+/// A computed `<percentage>` value.
+///
+/// FIXME(emilio): why is this in length.rs?
+#[derive(Clone, Copy, Debug, Default, PartialEq, HasViewportPercentage)]
+#[cfg_attr(feature = "servo", derive(Deserialize, HeapSizeOf, Serialize))]
+pub struct Percentage(pub CSSFloat);
+
+impl Percentage {
+    /// 0%
+    #[inline]
+    pub fn zero() -> Self {
+        Percentage(0.)
+    }
+
+    /// 100%
+    #[inline]
+    pub fn hundred() -> Self {
+        Percentage(1.)
+    }
+}
+
+impl ToCss for Percentage {
+    fn to_css<W>(&self, dest: &mut W) -> fmt::Result where W: fmt::Write {
+        write!(dest, "{}%", self.0 * 100.)
+    }
+}
+
+impl ToComputedValue for specified::Percentage {
+    type ComputedValue = Percentage;
+
+    #[inline]
+    fn to_computed_value(&self, _: &Context) -> Percentage {
+        Percentage(self.get())
+    }
+
+    #[inline]
+    fn from_computed_value(computed: &Percentage) -> Self {
+        specified::Percentage::new(computed.0)
+    }
+}
+
 impl ToComputedValue for specified::NoCalcLength {
     type ComputedValue = Au;
 
     #[inline]
     fn to_computed_value(&self, context: &Context) -> Au {
         match *self {
             specified::NoCalcLength::Absolute(length) =>
                 length.to_computed_value(context),
--- a/servo/components/style/values/computed/mod.rs
+++ b/servo/components/style/values/computed/mod.rs
@@ -33,21 +33,21 @@ pub use self::effects::{BoxShadow, Filte
 pub use self::flex::FlexBasis;
 pub use self::image::{Gradient, GradientItem, Image, ImageLayer, LineDirection, MozImageRect};
 #[cfg(feature = "gecko")]
 pub use self::gecko::ScrollSnapPoint;
 pub use self::rect::LengthOrNumberRect;
 pub use super::{Auto, Either, None_};
 #[cfg(feature = "gecko")]
 pub use super::specified::{AlignItems, AlignJustifyContent, AlignJustifySelf, JustifyItems};
-pub use super::specified::{BorderStyle, Percentage, UrlOrNone};
+pub use super::specified::{BorderStyle, UrlOrNone};
 pub use super::generics::grid::GridLine;
 pub use super::specified::url::SpecifiedUrl;
 pub use self::length::{CalcLengthOrPercentage, Length, LengthOrNone, LengthOrNumber, LengthOrPercentage};
-pub use self::length::{LengthOrPercentageOrAuto, LengthOrPercentageOrNone, MaxLength, MozLength};
+pub use self::length::{LengthOrPercentageOrAuto, LengthOrPercentageOrNone, MaxLength, MozLength, Percentage};
 pub use self::position::Position;
 pub use self::text::{InitialLetter, LetterSpacing, LineHeight, WordSpacing};
 pub use self::transform::{TimingFunction, TransformOrigin};
 
 pub mod background;
 pub mod basic_shape;
 pub mod border;
 pub mod color;
--- a/servo/components/style/values/computed/position.rs
+++ b/servo/components/style/values/computed/position.rs
@@ -4,19 +4,18 @@
 
 //! CSS handling for the computed value of
 //! [`position`][position] values.
 //!
 //! [position]: https://drafts.csswg.org/css-backgrounds-3/#position
 
 use std::fmt;
 use style_traits::ToCss;
-use values::computed::LengthOrPercentage;
+use values::computed::{LengthOrPercentage, Percentage};
 use values::generics::position::Position as GenericPosition;
-use values::specified::length::Percentage;
 
 /// The computed value of a CSS `<position>`
 pub type Position = GenericPosition<HorizontalPosition, VerticalPosition>;
 
 /// The computed value of a CSS horizontal position.
 pub type HorizontalPosition = LengthOrPercentage;
 
 /// The computed value of a CSS vertical position.
--- a/servo/components/style/values/computed/transform.rs
+++ b/servo/components/style/values/computed/transform.rs
@@ -1,19 +1,18 @@
 /* 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/. */
 
 //! Computed types for CSS values that are related to transformations.
 
 use properties::animated_properties::Animatable;
-use values::computed::{Length, LengthOrPercentage, Number};
+use values::computed::{Length, LengthOrPercentage, Number, Percentage};
 use values::generics::transform::TimingFunction as GenericTimingFunction;
 use values::generics::transform::TransformOrigin as GenericTransformOrigin;
-use values::specified::length::Percentage;
 
 /// The computed value of a CSS `<transform-origin>`
 pub type TransformOrigin = GenericTransformOrigin<LengthOrPercentage, LengthOrPercentage, Length>;
 
 /// A computed timing function.
 pub type TimingFunction = GenericTimingFunction<u32, Number>;
 
 impl TransformOrigin {
--- a/servo/components/style/values/generics/flex.rs
+++ b/servo/components/style/values/generics/flex.rs
@@ -1,15 +1,15 @@
 /* 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/. */
 
 //! Generic types for CSS values related to flexbox.
 
-use values::specified::Percentage;
+use values::computed::Percentage;
 
 /// A generic value for the `flex-basis` property.
 #[cfg_attr(feature = "servo", derive(HeapSizeOf))]
 #[derive(Clone, Copy, Debug, HasViewportPercentage, PartialEq, ToComputedValue, ToCss)]
 pub enum FlexBasis<LengthOrPercentage> {
     /// `auto`
     Auto,
     /// `content`
--- a/servo/components/style/values/specified/basic_shape.rs
+++ b/servo/components/style/values/specified/basic_shape.rs
@@ -7,24 +7,25 @@
 //!
 //! [basic-shape]: https://drafts.csswg.org/css-shapes/#typedef-basic-shape
 
 use cssparser::Parser;
 use parser::{Parse, ParserContext};
 use std::borrow::Cow;
 use std::fmt;
 use style_traits::{ToCss, ParseError, StyleParseError};
+use values::computed::Percentage;
 use values::generics::basic_shape::{Circle as GenericCircle};
 use values::generics::basic_shape::{ClippingShape as GenericClippingShape, Ellipse as GenericEllipse};
 use values::generics::basic_shape::{FillRule, BasicShape as GenericBasicShape};
 use values::generics::basic_shape::{FloatAreaShape as GenericFloatAreaShape, InsetRect as GenericInsetRect};
 use values::generics::basic_shape::{GeometryBox, ShapeBox, ShapeSource};
 use values::generics::basic_shape::{Polygon as GenericPolygon, ShapeRadius as GenericShapeRadius};
 use values::generics::rect::Rect;
-use values::specified::{LengthOrPercentage, Percentage};
+use values::specified::LengthOrPercentage;
 use values::specified::border::BorderRadius;
 use values::specified::position::{HorizontalPosition, Position, PositionComponent, Side, VerticalPosition};
 use values::specified::url::SpecifiedUrl;
 
 /// A specified clipping shape.
 pub type ClippingShape = GenericClippingShape<BasicShape>;
 
 /// A specified float area shape.
--- a/servo/components/style/values/specified/calc.rs
+++ b/servo/components/style/values/specified/calc.rs
@@ -9,19 +9,19 @@
 use app_units::Au;
 use cssparser::{Parser, Token, BasicParseError};
 use parser::ParserContext;
 use std::ascii::AsciiExt;
 use std::fmt;
 use style_traits::{HasViewportPercentage, ToCss, ParseError, StyleParseError};
 use style_traits::values::specified::AllowedLengthType;
 use values::{CSSInteger, CSSFloat};
+use values::computed;
 use values::specified::{Angle, Time};
-use values::specified::length::{FontRelativeLength, NoCalcLength};
-use values::specified::length::{Percentage, ViewportPercentageLength};
+use values::specified::length::{FontRelativeLength, NoCalcLength, ViewportPercentageLength};
 
 /// A node inside a `Calc` expression's AST.
 #[derive(Clone, Debug)]
 pub enum CalcNode {
     /// `<length>`
     Length(NoCalcLength),
     /// `<angle>`
     Angle(Angle),
@@ -47,16 +47,18 @@ pub enum CalcNode {
 #[derive(Clone, Copy, PartialEq)]
 pub enum CalcUnit {
     /// `<number>`
     Number,
     /// `<integer>`
     Integer,
     /// `<length>`
     Length,
+    /// `<percentage>`
+    Percentage,
     /// `<length> | <percentage>`
     LengthOrPercentage,
     /// `<angle>`
     Angle,
     /// `<time>`
     Time,
 }
 
@@ -70,17 +72,17 @@ pub struct CalcLengthOrPercentage {
     pub vw: Option<CSSFloat>,
     pub vh: Option<CSSFloat>,
     pub vmin: Option<CSSFloat>,
     pub vmax: Option<CSSFloat>,
     pub em: Option<CSSFloat>,
     pub ex: Option<CSSFloat>,
     pub ch: Option<CSSFloat>,
     pub rem: Option<CSSFloat>,
-    pub percentage: Option<Percentage>,
+    pub percentage: Option<computed::Percentage>,
     #[cfg(feature = "gecko")]
     pub mozmm: Option<CSSFloat>,
 }
 
 impl HasViewportPercentage for CalcLengthOrPercentage {
     fn has_viewport_percentage(&self) -> bool {
         self.vw.is_some() || self.vh.is_some() ||
         self.vmin.is_some() || self.vmax.is_some()
@@ -141,19 +143,18 @@ impl CalcNode {
     /// `<length>`, `<angle>`, `<time>`, `<percentage>`, according to
     /// `expected_unit`.
     ///
     /// May return a "complex" `CalcNode`, in the presence of a parenthesized
     /// expression, for example.
     fn parse_one<'i, 't>(
         context: &ParserContext,
         input: &mut Parser<'i, 't>,
-        expected_unit: CalcUnit)
-        -> Result<Self, ParseError<'i>>
-    {
+        expected_unit: CalcUnit
+    ) -> Result<Self, ParseError<'i>> {
         match (input.next()?, expected_unit) {
             (Token::Number { value, .. }, _) => Ok(CalcNode::Number(value)),
             (Token::Dimension { value, ref unit, .. }, CalcUnit::Length) |
             (Token::Dimension { value, ref unit, .. }, CalcUnit::LengthOrPercentage) => {
                 NoCalcLength::parse_dimension(context, value, unit)
                     .map(CalcNode::Length)
                     .map_err(|()| StyleParseError::UnspecifiedError.into())
             }
@@ -162,17 +163,18 @@ impl CalcNode {
                     .map(CalcNode::Angle)
                     .map_err(|()| StyleParseError::UnspecifiedError.into())
             }
             (Token::Dimension { value, ref unit, .. }, CalcUnit::Time) => {
                 Time::parse_dimension(value, unit, /* from_calc = */ true)
                     .map(CalcNode::Time)
                     .map_err(|()| StyleParseError::UnspecifiedError.into())
             }
-            (Token::Percentage { unit_value, .. }, CalcUnit::LengthOrPercentage) => {
+            (Token::Percentage { unit_value, .. }, CalcUnit::LengthOrPercentage) |
+            (Token::Percentage { unit_value, .. }, CalcUnit::Percentage) => {
                 Ok(CalcNode::Percentage(unit_value))
             }
             (Token::ParenthesisBlock, _) => {
                 input.parse_nested_block(|i| {
                     CalcNode::parse(context, i, expected_unit)
                 })
             }
             (Token::Function(ref name), _) if name.eq_ignore_ascii_case("calc") => {
@@ -278,29 +280,67 @@ impl CalcNode {
         let mut ret = CalcLengthOrPercentage {
             clamping_mode: clamping_mode,
             .. Default::default()
         };
         self.add_length_or_percentage_to(&mut ret, 1.0)?;
         Ok(ret)
     }
 
+    /// Tries to simplify this expression into a `<percentage>` value.
+    fn to_percentage(&self) -> Result<CSSFloat, ()> {
+        Ok(match *self {
+            CalcNode::Percentage(percentage) => percentage,
+            CalcNode::Sub(ref a, ref b) => {
+                a.to_percentage()? - b.to_percentage()?
+            }
+            CalcNode::Sum(ref a, ref b) => {
+                a.to_percentage()? + b.to_percentage()?
+            }
+            CalcNode::Mul(ref a, ref b) => {
+                match a.to_percentage() {
+                    Ok(lhs) => {
+                        let rhs = b.to_number()?;
+                        lhs * rhs
+                    }
+                    Err(..) => {
+                        let lhs = a.to_number()?;
+                        let rhs = b.to_percentage()?;
+                        lhs * rhs
+                    }
+                }
+            }
+            CalcNode::Div(ref a, ref b) => {
+                let lhs = a.to_percentage()?;
+                let rhs = b.to_number()?;
+                if rhs == 0. {
+                    return Err(())
+                }
+                lhs / rhs
+            }
+            CalcNode::Number(..) |
+            CalcNode::Length(..) |
+            CalcNode::Angle(..) |
+            CalcNode::Time(..) => return Err(()),
+        })
+    }
+
     /// Puts this `<length>` or `<percentage>` into `ret`, or error.
     ///
     /// `factor` is the sign or multiplicative factor to account for the sign
     /// (this allows adding and substracting into the return value).
     fn add_length_or_percentage_to(
         &self,
         ret: &mut CalcLengthOrPercentage,
         factor: CSSFloat)
         -> Result<(), ()>
     {
         match *self {
             CalcNode::Percentage(pct) => {
-                ret.percentage = Some(Percentage(
+                ret.percentage = Some(computed::Percentage(
                     ret.percentage.map_or(0., |p| p.0) + pct * factor,
                 ));
             }
             CalcNode::Length(ref l) => {
                 match *l {
                     NoCalcLength::Absolute(abs) => {
                         ret.absolute = Some(
                             ret.absolute.unwrap_or(Au(0)) +
@@ -490,74 +530,78 @@ impl CalcNode {
             CalcNode::Angle(..) |
             CalcNode::Time(..) => return Err(()),
         })
     }
 
     /// Convenience parsing function for integers.
     pub fn parse_integer<'i, 't>(
         context: &ParserContext,
-        input: &mut Parser<'i, 't>)
-        -> Result<CSSInteger, ParseError<'i>>
-    {
+        input: &mut Parser<'i, 't>
+    ) -> Result<CSSInteger, ParseError<'i>> {
         Self::parse(context, input, CalcUnit::Integer)?
             .to_number()
             .map(|n| n as CSSInteger)
             .map_err(|()| StyleParseError::UnspecifiedError.into())
     }
 
     /// Convenience parsing function for `<length> | <percentage>`.
     pub fn parse_length_or_percentage<'i, 't>(
         context: &ParserContext,
         input: &mut Parser<'i, 't>,
-        clamping_mode: AllowedLengthType)
-        -> Result<CalcLengthOrPercentage, ParseError<'i>>
-    {
+        clamping_mode: AllowedLengthType
+    ) -> Result<CalcLengthOrPercentage, ParseError<'i>> {
         Self::parse(context, input, CalcUnit::LengthOrPercentage)?
             .to_length_or_percentage(clamping_mode)
             .map_err(|()| StyleParseError::UnspecifiedError.into())
     }
 
+    /// Convenience parsing function for percentages.
+    pub fn parse_percentage<'i, 't>(
+        context: &ParserContext,
+        input: &mut Parser<'i, 't>
+    ) -> Result<CSSFloat, ParseError<'i>> {
+        Self::parse(context, input, CalcUnit::Percentage)?
+            .to_percentage()
+            .map_err(|()| StyleParseError::UnspecifiedError.into())
+    }
+
     /// Convenience parsing function for `<length>`.
     pub fn parse_length<'i, 't>(
         context: &ParserContext,
         input: &mut Parser<'i, 't>,
-        clamping_mode: AllowedLengthType)
-        -> Result<CalcLengthOrPercentage, ParseError<'i>>
-    {
+        clamping_mode: AllowedLengthType
+    ) -> Result<CalcLengthOrPercentage, ParseError<'i>> {
         Self::parse(context, input, CalcUnit::Length)?
             .to_length_or_percentage(clamping_mode)
             .map_err(|()| StyleParseError::UnspecifiedError.into())
     }
 
     /// Convenience parsing function for `<number>`.
     pub fn parse_number<'i, 't>(
         context: &ParserContext,
-        input: &mut Parser<'i, 't>)
-        -> Result<CSSFloat, ParseError<'i>>
-    {
+        input: &mut Parser<'i, 't>
+    ) -> Result<CSSFloat, ParseError<'i>> {
         Self::parse(context, input, CalcUnit::Number)?
             .to_number()
             .map_err(|()| StyleParseError::UnspecifiedError.into())
     }
 
     /// Convenience parsing function for `<angle>`.
     pub fn parse_angle<'i, 't>(
         context: &ParserContext,
-        input: &mut Parser<'i, 't>)
-        -> Result<Angle, ParseError<'i>>
-    {
+        input: &mut Parser<'i, 't>
+    ) -> Result<Angle, ParseError<'i>> {
         Self::parse(context, input, CalcUnit::Angle)?
             .to_angle()
             .map_err(|()| StyleParseError::UnspecifiedError.into())
     }
 
     /// Convenience parsing function for `<time>`.
     pub fn parse_time<'i, 't>(
         context: &ParserContext,
-        input: &mut Parser<'i, 't>)
-        -> Result<Time, ParseError<'i>>
-    {
+        input: &mut Parser<'i, 't>
+    ) -> Result<Time, ParseError<'i>> {
         Self::parse(context, input, CalcUnit::Time)?
             .to_time()
             .map_err(|()| StyleParseError::UnspecifiedError.into())
     }
 }
--- a/servo/components/style/values/specified/effects.rs
+++ b/servo/components/style/values/specified/effects.rs
@@ -1,26 +1,26 @@
 /* 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/. */
 
 //! Specified types for CSS values related to effects.
 
-use cssparser::{BasicParseError, Parser, Token};
+use cssparser::Parser;
 use parser::{Parse, ParserContext};
 use style_traits::{ParseError, StyleParseError};
 #[cfg(not(feature = "gecko"))]
 use values::Impossible;
 use values::computed::{Context, Number as ComputedNumber, ToComputedValue};
 use values::computed::effects::BoxShadow as ComputedBoxShadow;
 use values::computed::effects::SimpleShadow as ComputedSimpleShadow;
 use values::generics::effects::BoxShadow as GenericBoxShadow;
 use values::generics::effects::Filter as GenericFilter;
 use values::generics::effects::SimpleShadow as GenericSimpleShadow;
-use values::specified::{Angle, Percentage};
+use values::specified::{Angle, NumberOrPercentage};
 use values::specified::color::Color;
 use values::specified::length::Length;
 #[cfg(feature = "gecko")]
 use values::specified::url::SpecifiedUrl;
 
 /// A specified value for a single shadow of the `box-shadow` property.
 pub type BoxShadow = GenericBoxShadow<Option<Color>, Length, Option<Length>>;
 
@@ -28,25 +28,46 @@ pub type BoxShadow = GenericBoxShadow<Op
 #[cfg(feature = "gecko")]
 pub type Filter = GenericFilter<Angle, Factor, Length, SimpleShadow>;
 
 /// A specified value for a single `filter`.
 #[cfg(not(feature = "gecko"))]
 pub type Filter = GenericFilter<Angle, Factor, Length, Impossible>;
 
 /// A value for the `<factor>` parts in `Filter`.
-///
-/// FIXME: Should be `NumberOrPercentage`, but Gecko doesn't support that yet.
+#[derive(Clone, Debug, HasViewportPercentage, PartialEq, ToCss)]
 #[cfg_attr(feature = "servo", derive(HeapSizeOf))]
-#[derive(Clone, Debug, HasViewportPercentage, PartialEq, ToCss)]
-pub enum Factor {
-    /// Literal number.
-    Number(ComputedNumber),
-    /// Literal percentage.
-    Percentage(Percentage),
+pub struct Factor(NumberOrPercentage);
+
+impl Parse for Factor {
+    #[inline]
+    fn parse<'i, 't>(
+        context: &ParserContext,
+        input: &mut Parser<'i, 't>
+    ) -> Result<Self, ParseError<'i>> {
+        NumberOrPercentage::parse_non_negative(context, input).map(Factor)
+    }
+}
+
+impl ToComputedValue for Factor {
+    type ComputedValue = ComputedNumber;
+
+    #[inline]
+    fn to_computed_value(&self, context: &Context) -> Self::ComputedValue {
+        use values::computed::NumberOrPercentage;
+        match self.0.to_computed_value(context) {
+            NumberOrPercentage::Number(n) => n,
+            NumberOrPercentage::Percentage(p) => p.0,
+        }
+    }
+
+    #[inline]
+    fn from_computed_value(computed: &Self::ComputedValue) -> Self {
+        Factor(NumberOrPercentage::Number(ToComputedValue::from_computed_value(computed)))
+    }
 }
 
 /// A specified value for the `drop-shadow()` filter.
 pub type SimpleShadow = GenericSimpleShadow<Option<Color>, Length, Option<Length>>;
 
 impl Parse for BoxShadow {
     fn parse<'i, 't>(
         context: &ParserContext,
@@ -151,53 +172,16 @@ impl Parse for Filter {
                 "saturate" => Ok(GenericFilter::Saturate(Factor::parse(context, i)?)),
                 "sepia" => Ok(GenericFilter::Sepia(Factor::parse(context, i)?)),
                 "drop-shadow" => Ok(GenericFilter::DropShadow(Parse::parse(context, i)?)),
             }
         })
     }
 }
 
-impl Parse for Factor {
-    #[inline]
-    fn parse<'i, 't>(
-        _context: &ParserContext,
-        input: &mut Parser<'i, 't>
-    ) -> Result<Self, ParseError<'i>> {
-        match input.next()? {
-            Token::Number { value, .. } if value.is_sign_positive() => {
-                Ok(Factor::Number(value))
-            },
-            Token::Percentage { unit_value, .. } if unit_value.is_sign_positive() => {
-                Ok(Factor::Percentage(Percentage(unit_value)))
-            },
-            other => Err(BasicParseError::UnexpectedToken(other).into()),
-        }
-    }
-}
-
-impl ToComputedValue for Factor {
-    /// This should actually be `ComputedNumberOrPercentage`, but layout uses
-    /// `computed::effects::Filter` directly in `StackingContext`.
-    type ComputedValue = ComputedNumber;
-
-    #[inline]
-    fn to_computed_value(&self, _context: &Context) -> Self::ComputedValue {
-        match *self {
-            Factor::Number(number) => number,
-            Factor::Percentage(percentage) => percentage.0,
-        }
-    }
-
-    #[inline]
-    fn from_computed_value(computed: &Self::ComputedValue) -> Self {
-        Factor::Number(*computed)
-    }
-}
-
 impl Parse for SimpleShadow {
     #[inline]
     fn parse<'i, 't>(
         context: &ParserContext,
         input: &mut Parser<'i, 't>
     ) -> Result<Self, ParseError<'i>> {
         let color = input.try(|i| Color::parse(context, i)).ok();
         let horizontal = Length::parse(context, input)?;
--- a/servo/components/style/values/specified/image.rs
+++ b/servo/components/style/values/specified/image.rs
@@ -325,20 +325,20 @@ impl Gradient {
                     Ok(Self::new(x, y))
                 })
             }
         }
 
         impl<S: Side> From<Component<S>> for NumberOrPercentage {
             fn from(component: Component<S>) -> Self {
                 match component {
-                    Component::Center => NumberOrPercentage::Percentage(Percentage(0.5)),
+                    Component::Center => NumberOrPercentage::Percentage(Percentage::new(0.5)),
                     Component::Number(number) => number,
                     Component::Side(side) => {
-                        let p = Percentage(if side.is_start() { 0. } else { 1. });
+                        let p = if side.is_start() { Percentage::zero() } else { Percentage::hundred() };
                         NumberOrPercentage::Percentage(p)
                     },
                 }
             }
         }
 
         impl<S: Side> From<Component<S>> for PositionComponent<S> {
             fn from(component: Component<S>) -> Self {
@@ -358,17 +358,17 @@ impl Gradient {
                 }
             }
         }
 
         impl<S: Copy + Side> Component<S> {
             fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
                 match (NumberOrPercentage::from(*self), NumberOrPercentage::from(*other)) {
                     (NumberOrPercentage::Percentage(a), NumberOrPercentage::Percentage(b)) => {
-                        a.0.partial_cmp(&b.0)
+                        a.get().partial_cmp(&b.get())
                     },
                     (NumberOrPercentage::Number(a), NumberOrPercentage::Number(b)) => {
                         a.value.partial_cmp(&b.value)
                     },
                     (_, _) => {
                         None
                     }
                 }
@@ -439,51 +439,51 @@ impl Gradient {
         let mut items = input.try(|i| {
             i.expect_comma()?;
             i.parse_comma_separated(|i| {
                 let function = i.expect_function()?;
                 let (color, mut p) = i.parse_nested_block(|i| {
                     let p = match_ignore_ascii_case! { &function,
                         "color-stop" => {
                             let p = match NumberOrPercentage::parse(context, i)? {
-                                NumberOrPercentage::Number(number) => number.value,
-                                NumberOrPercentage::Percentage(p) => p.0,
+                                NumberOrPercentage::Number(number) => Percentage::new(number.value),
+                                NumberOrPercentage::Percentage(p) => p,
                             };
                             i.expect_comma()?;
                             p
                         },
-                        "from" => 0.,
-                        "to" => 1.,
+                        "from" => Percentage::zero(),
+                        "to" => Percentage::hundred(),
                         _ => return Err(StyleParseError::UnexpectedFunction(function.clone()).into()),
                     };
                     let color = Color::parse(context, i)?;
                     if color == Color::CurrentColor {
                         return Err(StyleParseError::UnspecifiedError.into());
                     }
                     Ok((color.into(), p))
                 })?;
                 if reverse_stops {
-                    p = 1. - p;
+                    p.reverse();
                 }
                 Ok(GenericGradientItem::ColorStop(GenericColorStop {
                     color: color,
-                    position: Some(LengthOrPercentage::Percentage(Percentage(p))),
+                    position: Some(p.into()),
                 }))
             })
         }).unwrap_or(vec![]);
 
         if items.is_empty() {
             items = vec![
                 GenericGradientItem::ColorStop(GenericColorStop {
                     color: Color::transparent().into(),
-                    position: Some(Percentage(0.).into()),
+                    position: Some(Percentage::zero().into()),
                 }),
                 GenericGradientItem::ColorStop(GenericColorStop {
                     color: Color::transparent().into(),
-                    position: Some(Percentage(1.).into()),
+                    position: Some(Percentage::hundred().into()),
                 }),
             ];
         } else if items.len() == 1 {
             let first = items[0].clone();
             items.push(first);
         } else {
             items.sort_by(|a, b| {
                 match (a, b) {
--- a/servo/components/style/values/specified/length.rs
+++ b/servo/components/style/values/specified/length.rs
@@ -15,17 +15,17 @@ use std::{cmp, fmt, mem};
 use std::ascii::AsciiExt;
 use std::ops::Mul;
 use style_traits::{HasViewportPercentage, ToCss, ParseError, StyleParseError};
 use style_traits::values::specified::{AllowedLengthType, AllowedNumericType};
 use stylesheets::CssRuleType;
 use super::{AllowQuirks, Number, ToComputedValue};
 use values::{Auto, CSSFloat, Either, FONT_MEDIUM_PX, None_, Normal};
 use values::ExtremumLength;
-use values::computed::{ComputedValueAsSpecified, Context};
+use values::computed::{self, Context};
 use values::specified::calc::CalcNode;
 
 pub use values::specified::calc::CalcLengthOrPercentage;
 pub use super::image::{ColorStop, EndingShape as GradientEndingShape, Gradient};
 pub use super::image::{GradientKind, Image};
 
 /// Number of app units per pixel
 pub const AU_PER_PX: CSSFloat = 60.;
@@ -696,90 +696,139 @@ impl<T: Parse> Either<Length, T> {
         if let Ok(v) = input.try(|input| T::parse(context, input)) {
             return Ok(Either::Second(v));
         }
         Length::parse_internal(context, input, AllowedLengthType::NonNegative, AllowQuirks::No).map(Either::First)
     }
 }
 
 /// A percentage value.
-///
-/// [0 .. 100%] maps to [0.0 .. 1.0]
-///
-/// FIXME(emilio): There's no standard property that requires a `<percentage>`
-/// without requiring also a `<length>`. If such a property existed, we'd need
-/// to add special handling for `calc()` and percentages in here in the same way
-/// as for `Angle` and `Time`, but the lack of this this is otherwise
-/// undistinguishable (we handle it correctly from `CalcLengthOrPercentage`).
-///
-/// As of today, only `-moz-image-rect` supports percentages without length.
-/// This is not a regression, and that's a non-standard extension anyway, so I'm
-/// not implementing it for now.
-#[derive(Clone, Copy, Debug, Default, HasViewportPercentage, PartialEq)]
-#[cfg_attr(feature = "servo", derive(Deserialize, HeapSizeOf, Serialize))]
-pub struct Percentage(pub CSSFloat);
+#[derive(Clone, Copy, Debug, Default, PartialEq)]
+#[cfg_attr(feature = "servo", derive(HeapSizeOf))]
+pub struct Percentage {
+    /// The percentage value as a float.
+    ///
+    /// [0 .. 100%] maps to [0.0 .. 1.0]
+    value: CSSFloat,
+    /// If this percentage came from a calc() expression, this tells how
+    /// clamping should be done on the value.
+    calc_clamping_mode: Option<AllowedNumericType>,
+}
+
+no_viewport_percentage!(Percentage);
 
 impl ToCss for Percentage {
     fn to_css<W>(&self, dest: &mut W) -> fmt::Result
         where W: fmt::Write,
     {
-        write!(dest, "{}%", self.0 * 100.)
+        if self.calc_clamping_mode.is_some() {
+            dest.write_str("calc(")?;
+        }
+
+        write!(dest, "{}%", self.value * 100.)?;
+
+        if self.calc_clamping_mode.is_some() {
+            dest.write_str(")")?;
+        }
+        Ok(())
     }
 }
 
 impl Percentage {
+    /// Create a percentage from a numeric value.
+    pub fn new(value: CSSFloat) -> Self {
+        Self {
+            value,
+            calc_clamping_mode: None,
+        }
+    }
+
+    /// Get the underlying value for this float.
+    pub fn get(&self) -> CSSFloat {
+        self.calc_clamping_mode.map_or(self.value, |mode| mode.clamp(self.value))
+    }
+
+    /// Reverse this percentage, preserving calc-ness.
+    ///
+    /// For example: If it was 20%, convert it into 80%.
+    pub fn reverse(&mut self) {
+        let new_value = 1. - self.value;
+        self.value = new_value;
+    }
+
+
     /// Parse a specific kind of percentage.
-    pub fn parse_with_clamping_mode<'i, 't>(context: &ParserContext,
-                                            input: &mut Parser<'i, 't>,
-                                            num_context: AllowedNumericType)
-                                            -> Result<Self, ParseError<'i>> {
+    pub fn parse_with_clamping_mode<'i, 't>(
+        context: &ParserContext,
+        input: &mut Parser<'i, 't>,
+        num_context: AllowedNumericType,
+    ) -> Result<Self, ParseError<'i>> {
         match input.next()? {
             Token::Percentage { unit_value, .. } if num_context.is_ok(context.parsing_mode, unit_value) => {
-                Ok(Percentage(unit_value))
+                Ok(Percentage::new(unit_value))
+            }
+            Token::Function(ref name) if name.eq_ignore_ascii_case("calc") => {
+                let result = input.parse_nested_block(|i| {
+                    CalcNode::parse_percentage(context, i)
+                })?;
+
+                // TODO(emilio): -moz-image-rect is the only thing that uses
+                // the clamping mode... I guess we could disallow it...
+                Ok(Percentage {
+                    value: result,
+                    calc_clamping_mode: Some(num_context),
+                })
             }
             t => Err(BasicParseError::UnexpectedToken(t).into())
         }
     }
 
     /// Parses a percentage token, but rejects it if it's negative.
     pub fn parse_non_negative<'i, 't>(context: &ParserContext,
                                       input: &mut Parser<'i, 't>)
                                       -> Result<Self, ParseError<'i>> {
         Self::parse_with_clamping_mode(context, input, AllowedNumericType::NonNegative)
     }
 
     /// 0%
     #[inline]
     pub fn zero() -> Self {
-        Percentage(0.)
+        Percentage {
+            value: 0.,
+            calc_clamping_mode: None,
+        }
     }
 
     /// 100%
     #[inline]
     pub fn hundred() -> Self {
-        Percentage(1.)
+        Percentage {
+            value: 1.,
+            calc_clamping_mode: None,
+        }
     }
 }
 
 impl Parse for Percentage {
     #[inline]
-    fn parse<'i, 't>(context: &ParserContext, input: &mut Parser<'i, 't>) -> Result<Self, ParseError<'i>> {
+    fn parse<'i, 't>(
+        context: &ParserContext,
+        input: &mut Parser<'i, 't>
+    ) -> Result<Self, ParseError<'i>> {
         Self::parse_with_clamping_mode(context, input, AllowedNumericType::All)
     }
 }
 
-impl ComputedValueAsSpecified for Percentage {}
-
 /// A length or a percentage value.
 #[allow(missing_docs)]
 #[cfg_attr(feature = "servo", derive(HeapSizeOf))]
 #[derive(Clone, Debug, HasViewportPercentage, PartialEq, ToCss)]
 pub enum LengthOrPercentage {
     Length(NoCalcLength),
-    Percentage(Percentage),
+    Percentage(computed::Percentage),
     Calc(Box<CalcLengthOrPercentage>),
 }
 
 impl From<Length> for LengthOrPercentage {
     fn from(len: Length) -> LengthOrPercentage {
         match len {
             Length::NoCalc(l) => LengthOrPercentage::Length(l),
             Length::Calc(l) => LengthOrPercentage::Calc(l),
@@ -792,16 +841,30 @@ impl From<NoCalcLength> for LengthOrPerc
     fn from(len: NoCalcLength) -> Self {
         LengthOrPercentage::Length(len)
     }
 }
 
 impl From<Percentage> for LengthOrPercentage {
     #[inline]
     fn from(pc: Percentage) -> Self {
+        if pc.calc_clamping_mode.is_some() {
+            LengthOrPercentage::Calc(Box::new(CalcLengthOrPercentage {
+                percentage: Some(computed::Percentage(pc.get())),
+                .. Default::default()
+            }))
+        } else {
+            LengthOrPercentage::Percentage(computed::Percentage(pc.get()))
+        }
+    }
+}
+
+impl From<computed::Percentage> for LengthOrPercentage {
+    #[inline]
+    fn from(pc: computed::Percentage) -> Self {
         LengthOrPercentage::Percentage(pc)
     }
 }
 
 impl LengthOrPercentage {
     #[inline]
     /// Returns a `zero` length.
     pub fn zero() -> LengthOrPercentage {
@@ -815,17 +878,17 @@ impl LengthOrPercentage {
                               -> Result<LengthOrPercentage, ParseError<'i>>
     {
         let token = input.next()?;
         match token {
             Token::Dimension { value, ref unit, .. } if num_context.is_ok(context.parsing_mode, value) => {
                 NoCalcLength::parse_dimension(context, value, unit).map(LengthOrPercentage::Length)
             }
             Token::Percentage { unit_value, .. } if num_context.is_ok(context.parsing_mode, unit_value) => {
-                return Ok(LengthOrPercentage::Percentage(Percentage(unit_value)))
+                return Ok(LengthOrPercentage::Percentage(computed::Percentage(unit_value)))
             }
             Token::Number { value, .. } if num_context.is_ok(context.parsing_mode, value) => {
                 if value != 0. &&
                    !context.parsing_mode.allows_unitless_lengths() &&
                    !allow_quirks.allowed(context.quirks_mode) {
                     Err(())
                 } else {
                     return Ok(LengthOrPercentage::Length(NoCalcLength::from_px(value)))
@@ -920,48 +983,48 @@ impl LengthOrPercentage {
 }
 
 /// Either a `<length>`, a `<percentage>`, or the `auto` keyword.
 #[allow(missing_docs)]
 #[cfg_attr(feature = "servo", derive(HeapSizeOf))]
 #[derive(Clone, Debug, HasViewportPercentage, PartialEq, ToCss)]
 pub enum LengthOrPercentageOrAuto {
     Length(NoCalcLength),
-    Percentage(Percentage),
+    Percentage(computed::Percentage),
     Auto,
     Calc(Box<CalcLengthOrPercentage>),
 }
 
 impl From<NoCalcLength> for LengthOrPercentageOrAuto {
     #[inline]
     fn from(len: NoCalcLength) -> Self {
         LengthOrPercentageOrAuto::Length(len)
     }
 }
 
-impl From<Percentage> for LengthOrPercentageOrAuto {
+impl From<computed::Percentage> for LengthOrPercentageOrAuto {
     #[inline]
-    fn from(pc: Percentage) -> Self {
+    fn from(pc: computed::Percentage) -> Self {
         LengthOrPercentageOrAuto::Percentage(pc)
     }
 }
 
 impl LengthOrPercentageOrAuto {
     fn parse_internal<'i, 't>(context: &ParserContext,
                               input: &mut Parser<'i, 't>,
                               num_context: AllowedLengthType,
                               allow_quirks: AllowQuirks)
                               -> Result<Self, ParseError<'i>> {
         let token = input.next()?;
         match token {
             Token::Dimension { value, ref unit, .. } if num_context.is_ok(context.parsing_mode, value) => {
                 NoCalcLength::parse_dimension(context, value, unit).map(LengthOrPercentageOrAuto::Length)
             }
             Token::Percentage { unit_value, .. } if num_context.is_ok(context.parsing_mode, unit_value) => {
-                Ok(LengthOrPercentageOrAuto::Percentage(Percentage(unit_value)))
+                Ok(LengthOrPercentageOrAuto::Percentage(computed::Percentage(unit_value)))
             }
             Token::Number { value, .. } if num_context.is_ok(context.parsing_mode, value) => {
                 if value != 0. &&
                    !context.parsing_mode.allows_unitless_lengths() &&
                    !allow_quirks.allowed(context.quirks_mode) {
                     return Err(StyleParseError::UnspecifiedError.into())
                 }
                 Ok(LengthOrPercentageOrAuto::Length(
@@ -1004,17 +1067,17 @@ impl LengthOrPercentageOrAuto {
 
     /// Returns a value representing a `0` length.
     pub fn zero() -> Self {
         LengthOrPercentageOrAuto::Length(NoCalcLength::zero())
     }
 
     /// Returns a value representing `0%`.
     pub fn zero_percent() -> Self {
-        LengthOrPercentageOrAuto::Percentage(Percentage::zero())
+        LengthOrPercentageOrAuto::Percentage(computed::Percentage::zero())
     }
 }
 
 impl Parse for LengthOrPercentageOrAuto {
     #[inline]
     fn parse<'i, 't>(context: &ParserContext, input: &mut Parser<'i, 't>) -> Result<Self, ParseError<'i>> {
         Self::parse_quirky(context, input, AllowQuirks::No)
     }
@@ -1032,17 +1095,17 @@ impl LengthOrPercentageOrAuto {
 }
 
 /// Either a `<length>`, a `<percentage>`, or the `none` keyword.
 #[cfg_attr(feature = "servo", derive(HeapSizeOf))]
 #[derive(Clone, Debug, HasViewportPercentage, PartialEq, ToCss)]
 #[allow(missing_docs)]
 pub enum LengthOrPercentageOrNone {
     Length(NoCalcLength),
-    Percentage(Percentage),
+    Percentage(computed::Percentage),
     Calc(Box<CalcLengthOrPercentage>),
     None,
 }
 
 impl LengthOrPercentageOrNone {
     fn parse_internal<'i, 't>(context: &ParserContext,
                               input: &mut Parser<'i, 't>,
                               num_context: AllowedLengthType,
@@ -1050,17 +1113,17 @@ impl LengthOrPercentageOrNone {
                               -> Result<LengthOrPercentageOrNone, ParseError<'i>>
     {
         let token = input.next()?;
         match token {
             Token::Dimension { value, ref unit, .. } if num_context.is_ok(context.parsing_mode, value) => {
                 NoCalcLength::parse_dimension(context, value, unit).map(LengthOrPercentageOrNone::Length)
             }
             Token::Percentage { unit_value, .. } if num_context.is_ok(context.parsing_mode, unit_value) => {
-                Ok(LengthOrPercentageOrNone::Percentage(Percentage(unit_value)))
+                Ok(LengthOrPercentageOrNone::Percentage(computed::Percentage(unit_value)))
             }
             Token::Number { value, .. } if num_context.is_ok(context.parsing_mode, value) => {
                 if value != 0. && !context.parsing_mode.allows_unitless_lengths() &&
                    !allow_quirks.allowed(context.quirks_mode) {
                     return Err(StyleParseError::UnspecifiedError.into())
                 }
                 Ok(LengthOrPercentageOrNone::Length(
                     NoCalcLength::Absolute(AbsoluteLength::Px(value))
--- a/servo/components/style/values/specified/mod.rs
+++ b/servo/components/style/values/specified/mod.rs
@@ -499,33 +499,37 @@ impl ToCss for Number {
         self.value.to_css(dest)?;
         if self.calc_clamping_mode.is_some() {
             dest.write_str(")")?;
         }
         Ok(())
     }
 }
 
-/// <number-percentage>
+/// <number> | <percentage>
+///
 /// Accepts only non-negative numbers.
+///
+/// FIXME(emilio): Should probably use Either.
 #[allow(missing_docs)]
 #[cfg_attr(feature = "servo", derive(HeapSizeOf))]
 #[derive(Clone, Copy, Debug, PartialEq, ToCss)]
 pub enum NumberOrPercentage {
     Percentage(Percentage),
     Number(Number),
 }
 
 no_viewport_percentage!(NumberOrPercentage);
 
 impl NumberOrPercentage {
-    fn parse_with_clamping_mode<'i, 't>(context: &ParserContext,
-                                        input: &mut Parser<'i, 't>,
-                                        type_: AllowedNumericType)
-                                        -> Result<Self, ParseError<'i>> {
+    fn parse_with_clamping_mode<'i, 't>(
+        context: &ParserContext,
+        input: &mut Parser<'i, 't>,
+        type_: AllowedNumericType
+    ) -> Result<Self, ParseError<'i>> {
         if let Ok(per) = input.try(|i| Percentage::parse_with_clamping_mode(context, i, type_)) {
             return Ok(NumberOrPercentage::Percentage(per));
         }
 
         parse_number_with_clamping_mode(context, input, type_).map(NumberOrPercentage::Number)
     }
 
     /// Parse a non-negative number or percentage.
--- a/servo/components/style/values/specified/position.rs
+++ b/servo/components/style/values/specified/position.rs
@@ -7,19 +7,19 @@
 //!
 //! [position]: https://drafts.csswg.org/css-backgrounds-3/#position
 
 use cssparser::Parser;
 use parser::{Parse, ParserContext};
 use std::fmt;
 use style_traits::{HasViewportPercentage, ToCss, ParseError};
 use values::computed::{CalcLengthOrPercentage, LengthOrPercentage as ComputedLengthOrPercentage};
-use values::computed::{Context, ToComputedValue};
+use values::computed::{Context, Percentage, ToComputedValue};
 use values::generics::position::Position as GenericPosition;
-use values::specified::{AllowQuirks, LengthOrPercentage, Percentage};
+use values::specified::{AllowQuirks, LengthOrPercentage};
 use values::specified::transform::OriginComponent;
 
 /// The specified value of a CSS `<position>`
 pub type Position = GenericPosition<HorizontalPosition, VerticalPosition>;
 
 /// The specified value of a horizontal position.
 pub type HorizontalPosition = PositionComponent<X>;
 
@@ -190,17 +190,17 @@ impl<S: Parse> PositionComponent<S> {
         let lop = input.try(|i| LengthOrPercentage::parse_quirky(context, i, allow_quirks)).ok();
         Ok(PositionComponent::Side(keyword, lop))
     }
 }
 
 impl<S> PositionComponent<S> {
     /// `0%`
     pub fn zero() -> Self {
-        PositionComponent::Length(LengthOrPercentage::Percentage(Percentage(0.)))
+        PositionComponent::Length(LengthOrPercentage::Percentage(Percentage::zero()))
     }
 }
 
 impl<S: Side> ToComputedValue for PositionComponent<S> {
     type ComputedValue = ComputedLengthOrPercentage;
 
     fn to_computed_value(&self, context: &Context) -> Self::ComputedValue {
         match *self {
@@ -209,17 +209,18 @@ impl<S: Side> ToComputedValue for Positi
             },
             PositionComponent::Side(ref keyword, None) => {
                 let p = Percentage(if keyword.is_start() { 0. } else { 1. });
                 ComputedLengthOrPercentage::Percentage(p)
             },
             PositionComponent::Side(ref keyword, Some(ref length)) if !keyword.is_start() => {
                 match length.to_computed_value(context) {
                     ComputedLengthOrPercentage::Length(length) => {
-                        ComputedLengthOrPercentage::Calc(CalcLengthOrPercentage::new(-length, Some(Percentage(1.0))))
+                        ComputedLengthOrPercentage::Calc(
+                            CalcLengthOrPercentage::new(-length, Some(Percentage::hundred())))
                     },
                     ComputedLengthOrPercentage::Percentage(p) => {
                         ComputedLengthOrPercentage::Percentage(Percentage(1.0 - p.0))
                     },
                     ComputedLengthOrPercentage::Calc(calc) => {
                         let p = Percentage(1. - calc.percentage.map_or(0., |p| p.0));
                         ComputedLengthOrPercentage::Calc(CalcLengthOrPercentage::new(-calc.unclamped_length(), Some(p)))
                     },
--- a/servo/components/style/values/specified/transform.rs
+++ b/servo/components/style/values/specified/transform.rs
@@ -4,22 +4,23 @@
 
 //! Specified types for CSS values that are related to transformations.
 
 use cssparser::Parser;
 use euclid::Point2D;
 use parser::{Parse, ParserContext};
 use selectors::parser::SelectorParseError;
 use style_traits::{ParseError, StyleParseError};
-use values::computed::{LengthOrPercentage as ComputedLengthOrPercentage, Context, ToComputedValue};
+use values::computed::{Context, LengthOrPercentage as ComputedLengthOrPercentage};
+use values::computed::{Percentage as ComputedPercentage, ToComputedValue};
 use values::computed::transform::TimingFunction as ComputedTimingFunction;
 use values::generics::transform::{StepPosition, TimingFunction as GenericTimingFunction};
 use values::generics::transform::{TimingKeyword, TransformOrigin as GenericTransformOrigin};
 use values::specified::{Integer, Number};
-use values::specified::length::{Length, LengthOrPercentage, Percentage};
+use values::specified::length::{Length, LengthOrPercentage};
 use values::specified::position::{Side, X, Y};
 
 /// The specified value of a CSS `<transform-origin>`
 pub type TransformOrigin = GenericTransformOrigin<OriginComponent<X>, OriginComponent<Y>, Length>;
 
 /// The specified value of a component of a CSS `<transform-origin>`.
 #[cfg_attr(feature = "servo", derive(HeapSizeOf))]
 #[derive(Clone, Debug, HasViewportPercentage, PartialEq, ToCss)]
@@ -102,37 +103,37 @@ impl<S> Parse for OriginComponent<S>
 impl<S> ToComputedValue for OriginComponent<S>
     where S: Side,
 {
     type ComputedValue = ComputedLengthOrPercentage;
 
     fn to_computed_value(&self, context: &Context) -> Self::ComputedValue {
         match *self {
             OriginComponent::Center => {
-                ComputedLengthOrPercentage::Percentage(Percentage(0.5))
+                ComputedLengthOrPercentage::Percentage(ComputedPercentage(0.5))
             },
             OriginComponent::Length(ref length) => {
                 length.to_computed_value(context)
             },
             OriginComponent::Side(ref keyword) => {
-                let p = Percentage(if keyword.is_start() { 0. } else { 1. });
+                let p = ComputedPercentage(if keyword.is_start() { 0. } else { 1. });
                 ComputedLengthOrPercentage::Percentage(p)
             },
         }
     }
 
     fn from_computed_value(computed: &Self::ComputedValue) -> Self {
         OriginComponent::Length(ToComputedValue::from_computed_value(computed))
     }
 }
 
 impl<S> OriginComponent<S> {
     /// `0%`
     pub fn zero() -> Self {
-        OriginComponent::Length(LengthOrPercentage::Percentage(Percentage(0.)))
+        OriginComponent::Length(LengthOrPercentage::Percentage(ComputedPercentage::zero()))
     }
 }
 
 #[cfg(feature = "gecko")]
 #[inline]
 fn allow_frames_timing() -> bool {
     use gecko_bindings::bindings;
     unsafe { bindings::Gecko_IsFramesTimingEnabled() }
--- a/servo/components/style_traits/lib.rs
+++ b/servo/components/style_traits/lib.rs
@@ -86,17 +86,17 @@ pub enum StyleParseError<'i> {
     BadStringInDeclarationValueBlock(CompactCowStr<'i>),
     /// Unexpected closing parenthesis in a DVB.
     UnbalancedCloseParenthesisInDeclarationValueBlock,
     /// Unexpected closing bracket in a DVB.
     UnbalancedCloseSquareBracketInDeclarationValueBlock,
     /// Unexpected closing curly bracket in a DVB.
     UnbalancedCloseCurlyBracketInDeclarationValueBlock,
     /// A property declaration parsing error.
-    PropertyDeclaration(PropertyDeclarationParseError),
+    PropertyDeclaration(PropertyDeclarationParseError<'i>),
     /// A property declaration value had input remaining after successfully parsing.
     PropertyDeclarationValueNotExhausted,
     /// An unexpected dimension token was encountered.
     UnexpectedDimension(CompactCowStr<'i>),
     /// A media query using a ranged expression with no value was encountered.
     RangedExpressionWithNoValue,
     /// A function was encountered that was not expected.
     UnexpectedFunction(CompactCowStr<'i>),
@@ -107,25 +107,25 @@ pub enum StyleParseError<'i> {
     /// Unexpected @charset rule encountered.
     UnexpectedCharsetRule,
     /// Unsupported @ rule
     UnsupportedAtRule(CompactCowStr<'i>),
     /// A placeholder for many sources of errors that require more specific variants.
     UnspecifiedError,
     /// An unexpected token was found within a namespace rule.
     UnexpectedTokenWithinNamespace(Token<'i>),
-    /// An unknown CSS property was encountered.
-    UnknownProperty(CompactCowStr<'i>),
 }
 
 /// The result of parsing a property declaration.
 #[derive(Eq, PartialEq, Clone, Debug)]
-pub enum PropertyDeclarationParseError {
+pub enum PropertyDeclarationParseError<'i> {
     /// The property declaration was for an unknown property.
-    UnknownProperty,
+    UnknownProperty(CompactCowStr<'i>),
+    /// An unknown vendor-specific identifier was encountered.
+    UnknownVendorProperty,
     /// The property declaration was for a disabled experimental property.
     ExperimentalProperty,
     /// The property declaration contained an invalid value.
     InvalidValue(String),
     /// The declaration contained an animation property, and we were parsing
     /// this as a keyframe block (so that property should be ignored).
     ///
     /// See: https://drafts.csswg.org/css-animations/#keyframes
@@ -135,18 +135,18 @@ pub enum PropertyDeclarationParseError {
 }
 
 impl<'a> From<StyleParseError<'a>> for ParseError<'a> {
     fn from(this: StyleParseError<'a>) -> Self {
         cssparser::ParseError::Custom(SelectorParseError::Custom(this))
     }
 }
 
-impl<'a> From<PropertyDeclarationParseError> for ParseError<'a> {
-    fn from(this: PropertyDeclarationParseError) -> Self {
+impl<'a> From<PropertyDeclarationParseError<'a>> for ParseError<'a> {
+    fn from(this: PropertyDeclarationParseError<'a>) -> Self {
         cssparser::ParseError::Custom(SelectorParseError::Custom(StyleParseError::PropertyDeclaration(this)))
     }
 }
 
 bitflags! {
     /// The mode to use when parsing values.
     pub flags ParsingMode: u8 {
         /// In CSS, lengths must have units, except for zero values, where the unit can be omitted.
--- a/servo/ports/cef/window.rs
+++ b/servo/ports/cef/window.rs
@@ -19,16 +19,17 @@ use wrappers::CefWrap;
 
 use compositing::compositor_thread::EventLoopWaker;
 use compositing::windowing::{WindowEvent, WindowMethods};
 use euclid::{Point2D, TypedPoint2D, TypedRect, Size2D, TypedSize2D, ScaleFactor};
 use gleam::gl;
 use msg::constellation_msg::{Key, KeyModifiers};
 use net_traits::net_error_list::NetError;
 use script_traits::{DevicePixel, LoadData};
+use servo::ipc_channel::ipc::IpcSender;
 use servo_geometry::DeviceIndependentPixel;
 use std::cell::RefCell;
 use std::ffi::CString;
 use std::os::raw::{c_char, c_void};
 use std::ptr;
 use std::rc::Rc;
 use std::sync::mpsc::{Sender, channel};
 use servo_url::ServoUrl;
@@ -484,18 +485,20 @@ impl WindowMethods for Window {
                        .get_client()
                        .get_render_handler()
                        .on_cursor_change(browser.clone(), cursor_handle,
                                          self.cursor_type_for_cursor(cursor), &info);
             }
         }
     }
 
-    fn allow_navigation(&self, _: ServoUrl) -> bool {
-        true
+    fn allow_navigation(&self, _: ServoUrl, response_chan: IpcSender<bool>) {
+        if let Err(e) = response_chan.send(true) {
+            warn!("Failed to send allow_navigation() response: {}", e);
+        };
     }
 
     fn supports_clipboard(&self) -> bool {
         false
     }
 }
 
 #[cfg(target_os="macos")]
--- a/servo/ports/geckolib/error_reporter.rs
+++ b/servo/ports/geckolib/error_reporter.rs
@@ -228,17 +228,18 @@ impl<'a> ErrorHelpers<'a> for Contextual
 
             (_, CssParseError::Custom(SelectorParseError::UnexpectedIdent(ident))) =>
                 ErrorString::Ident(ident),
 
             (_, CssParseError::Custom(SelectorParseError::ExpectedNamespace(namespace))) =>
                 ErrorString::Ident(namespace),
 
             (_, CssParseError::Custom(SelectorParseError::Custom(
-                StyleParseError::UnknownProperty(property)))) =>
+                StyleParseError::PropertyDeclaration(
+                    PropertyDeclarationParseError::UnknownProperty(property))))) =>
                 ErrorString::Ident(property),
 
             (_, CssParseError::Custom(SelectorParseError::Custom(
                 StyleParseError::UnexpectedTokenWithinNamespace(token)))) =>
                 ErrorString::UnexpectedToken(token),
 
             (s, _)  => ErrorString::Snippet(s)
         }
--- a/servo/ports/geckolib/glue.rs
+++ b/servo/ports/geckolib/glue.rs
@@ -2478,18 +2478,19 @@ pub extern "C" fn Servo_DeclarationBlock
 #[no_mangle]
 pub extern "C" fn Servo_DeclarationBlock_SetPercentValue(declarations:
                                                          RawServoDeclarationBlockBorrowed,
                                                          property: nsCSSPropertyID,
                                                          value: f32) {
     use style::properties::{PropertyDeclaration, LonghandId};
     use style::properties::longhands::height::SpecifiedValue as Height;
     use style::properties::longhands::width::SpecifiedValue as Width;
+    use style::values::computed::Percentage;
     use style::values::specified::MozLength;
-    use style::values::specified::length::{LengthOrPercentage, Percentage};
+    use style::values::specified::length::LengthOrPercentage;
 
     let long = get_longhand_from_id!(property);
     let pc = Percentage(value);
 
     let prop = match_wrap_declared! { long,
         Height => Height(MozLength::LengthOrPercentageOrAuto(pc.into())),
         Width => Width(MozLength::LengthOrPercentageOrAuto(pc.into())),
         MarginTop => pc.into(),
--- a/servo/ports/glutin/Cargo.toml
+++ b/servo/ports/glutin/Cargo.toml
@@ -8,16 +8,17 @@ license = "MPL-2.0"
 name = "glutin_app"
 path = "lib.rs"
 
 [dependencies]
 bitflags = "0.7"
 compositing = {path = "../../components/compositing"}
 euclid = "0.15"
 gleam = "0.4"
+libservo = {path = "../../components/servo"}
 log = "0.3.5"
 msg = {path = "../../components/msg"}
 net_traits = {path = "../../components/net_traits"}
 script_traits = {path = "../../components/script_traits"}
 servo-glutin = "0.11"
 servo_geometry = {path = "../../components/geometry"}
 servo_config = {path = "../../components/config"}
 servo_url = {path = "../../components/url"}
--- a/servo/ports/glutin/lib.rs
+++ b/servo/ports/glutin/lib.rs
@@ -12,16 +12,17 @@ extern crate compositing;
 extern crate euclid;
 extern crate gleam;
 extern crate glutin;
 #[macro_use] extern crate log;
 extern crate msg;
 extern crate net_traits;
 #[cfg(any(target_os = "linux", target_os = "macos"))] extern crate osmesa_sys;
 extern crate script_traits;
+extern crate servo;
 extern crate servo_config;
 extern crate servo_geometry;
 extern crate servo_url;
 extern crate style_traits;
 extern crate webrender_api;
 
 #[cfg(target_os = "windows")] extern crate winapi;
 #[cfg(target_os = "windows")] extern crate user32;
--- a/servo/ports/glutin/window.rs
+++ b/servo/ports/glutin/window.rs
@@ -20,16 +20,17 @@ use glutin::TouchPhase;
 #[cfg(target_os = "macos")]
 use glutin::os::macos::{ActivationPolicy, WindowBuilderExt};
 use msg::constellation_msg::{self, Key};
 use msg::constellation_msg::{ALT, CONTROL, KeyState, NONE, SHIFT, SUPER, TraversalDirection};
 use net_traits::net_error_list::NetError;
 #[cfg(any(target_os = "linux", target_os = "macos"))]
 use osmesa_sys;
 use script_traits::{DevicePixel, LoadData, TouchEventType, TouchpadPressurePhase};
+use servo::ipc_channel::ipc::IpcSender;
 use servo_config::opts;
 use servo_config::prefs::PREFS;
 use servo_config::resource_files;
 use servo_geometry::DeviceIndependentPixel;
 use servo_url::ServoUrl;
 use std::cell::{Cell, RefCell};
 #[cfg(any(target_os = "linux", target_os = "macos"))]
 use std::ffi::CString;
@@ -1295,18 +1296,20 @@ impl WindowMethods for Window {
             }
 
             _ => {
                 self.platform_handle_key(key, mods);
             }
         }
     }
 
-    fn allow_navigation(&self, _: ServoUrl) -> bool {
-        true
+    fn allow_navigation(&self, _: ServoUrl, response_chan: IpcSender<bool>) {
+        if let Err(e) = response_chan.send(true) {
+            warn!("Failed to send allow_navigation() response: {}", e);
+        };
     }
 
     fn supports_clipboard(&self) -> bool {
         true
     }
 }
 
 fn glutin_phase_to_touch_event_type(phase: TouchPhase) -> TouchEventType {
--- a/servo/python/servo/bootstrap_commands.py
+++ b/servo/python/servo/bootstrap_commands.py
@@ -130,21 +130,18 @@ class MachCommands(CommandBase):
                 print("Use |bootstrap-rust --force| to download again.")
                 continue
 
             if self.use_stable_rust():
                 std_url = ("https://static-rust-lang-org.s3.amazonaws.com/dist/rust-std-%s-%s.tar.gz"
                            % (version, target_triple))
                 tgz_file = install_dir + ('rust-std-%s-%s.tar.gz' % (version, target_triple))
             else:
-                tarball = "%s/rust-std-nightly-%s.tar.gz" % (version, target_triple)
-                base_url = "https://s3.amazonaws.com/rust-lang-ci/rustc-builds"
-                if not self.config["build"]["llvm-assertions"]:
-                    base_url += "-alt"
-                std_url = base_url + "/" + tarball
+                std_url = ("https://s3.amazonaws.com/rust-lang-ci/rustc-builds/%s/rust-std-nightly-%s.tar.gz"
+                           % (version, target_triple))
                 tgz_file = install_dir + ('rust-std-nightly-%s.tar.gz' % target_triple)
 
             download_file("Host rust library for target %s" % target_triple, std_url, tgz_file)
             print("Extracting Rust stdlib for target %s..." % target_triple)
             extract(tgz_file, install_dir)
             shutil.copytree(path.join(install_dir,
                                       "rust-std%s%s-%s" % (nightly_suffix, stable_version, target_triple),
                                       "rust-std-%s" % target_triple, "lib", "rustlib", target_triple),
--- a/servo/rust-commit-hash
+++ b/servo/rust-commit-hash
@@ -1,1 +1,1 @@
-f85579d4a2c342654f9b158fafd565eb159fdb59
+ab91c70cc69450bc69b76df0fd2faa82bda42d6b
--- a/servo/tests/unit/net/http_loader.rs
+++ b/servo/tests/unit/net/http_loader.rs
@@ -25,24 +25,28 @@ use msg::constellation_msg::TEST_PIPELIN
 use net::cookie::Cookie;
 use net::cookie_storage::CookieStorage;
 use net::resource_thread::AuthCacheEntry;
 use net::test::replace_host_table;
 use net_traits::{CookieSource, NetworkError};
 use net_traits::request::{Request, RequestInit, RequestMode, CredentialsMode, Destination};
 use net_traits::response::ResponseBody;
 use new_fetch_context;
-use servo_url::ServoUrl;
+use servo_url::{ServoUrl, ImmutableOrigin};
 use std::collections::HashMap;
 use std::io::{Read, Write};
 use std::str::FromStr;
 use std::sync::{Arc, Mutex, RwLock, mpsc};
 use std::sync::atomic::{AtomicBool, Ordering};
 use std::sync::mpsc::Receiver;
 
+fn mock_origin() -> ImmutableOrigin {
+    ServoUrl::parse("http://servo.org").unwrap().origin()
+}
+
 fn read_response(reader: &mut Read) -> String {
     let mut buf = vec![0; 1024];
     match reader.read(&mut buf) {
         Ok(len) if len > 0 => {
             unsafe { buf.set_len(len); }
             String::from_utf8(buf).unwrap()
         },
         Ok(_) => "".to_owned(),
@@ -133,63 +137,63 @@ fn test_check_default_headers_loaded_in_
 
     *expected_headers.lock().unwrap() = Some(headers.clone());
 
     // Testing for method.GET
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: url.clone().origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 
     // Testing for method.POST
     let mut post_headers = headers.clone();
     post_headers.set(ContentLength(0 as u64));
     let url_str = url.as_str();
     // request gets header "Origin: http://example.com" but expected_headers has
     // "Origin: http://example.com/" which do not match for equality so strip trailing '/'
     post_headers.set(Origin::from_str(&url_str[..url_str.len()-1]).unwrap());
     *expected_headers.lock().unwrap() = Some(post_headers);
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Post,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: url.clone().origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 
     let _ = server.close();
 }
 
 #[test]
 fn test_load_when_request_is_not_get_or_head_and_there_is_no_body_content_length_should_be_set_to_0() {
     let handler = move |request: HyperRequest, _: HyperResponse| {
         assert_eq!(request.headers.get::<ContentLength>(), Some(&ContentLength(0)));
     };
     let (mut server, url) = make_server(handler);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Post,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 
     let _ = server.close();
 }
 
 #[test]
 fn test_request_and_response_data_with_network_messages() {
     let handler = move |_: HyperRequest, mut response: HyperResponse| {
         response.headers_mut().set(Host { hostname: "foo.bar".to_owned(), port: None });
@@ -200,23 +204,23 @@ fn test_request_and_response_data_with_n
     let mut request_headers = Headers::new();
     request_headers.set(Host { hostname: "bar.foo".to_owned(), port: None });
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         headers: request_headers,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let (devtools_chan, devtools_port) = mpsc::channel();
     let response = fetch(&mut request, Some(devtools_chan));
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 
     let _ = server.close();
 
     // notification received from devtools
     let devhttprequest = expect_devtools_http_request(&devtools_port);
     let devhttpresponse = expect_devtools_http_response(&devtools_port);
 
     //Creating default headers for request
@@ -287,23 +291,23 @@ fn test_request_and_response_message_fro
         response.send(b"Yay!").unwrap();
     };
     let (mut server, url) = make_server(handler);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: None,
         .. RequestInit::default()
     });
     let (devtools_chan, devtools_port) = mpsc::channel();
     let response = fetch(&mut request, Some(devtools_chan));
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 
     let _ = server.close();
 
     // notification received from devtools
     assert!(devtools_port.try_recv().is_err());
 }
 
 #[test]
@@ -322,17 +326,16 @@ fn test_redirected_request_to_devtools()
         response.send(b"").unwrap();
     };
     let (mut pre_server, pre_url) = make_server(pre_handler);
 
     let mut request = Request::from_init(RequestInit {
         url: pre_url.clone(),
         method: Method::Post,
         destination: Destination::Document,
-        origin: pre_url.clone(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let (devtools_chan, devtools_port) = mpsc::channel();
     fetch(&mut request, Some(devtools_chan));
 
     let _ = pre_server.close();
     let _ = post_server.close();
@@ -370,17 +373,17 @@ fn test_load_when_redirecting_from_a_pos
         response.send(b"").unwrap();
     };
     let (mut pre_server, pre_url) = make_server(pre_handler);
 
     let mut request = Request::from_init(RequestInit {
         url: pre_url.clone(),
         method: Method::Post,
         destination: Destination::Document,
-        origin: pre_url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = pre_server.close();
     let _ = post_server.close();
 
@@ -398,26 +401,27 @@ fn test_load_should_decode_the_response_
     };
     let (mut server, url) = make_server(handler);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
-    assert_eq!(*response.body.lock().unwrap(),
+    let internal_response = response.internal_response.unwrap();
+    assert!(internal_response.status.unwrap().is_success());
+    assert_eq!(*internal_response.body.lock().unwrap(),
                ResponseBody::Done(b"Yay!".to_vec()));
 }
 
 #[test]
 fn test_load_should_decode_the_response_as_gzip_when_response_headers_have_content_encoding_gzip() {
     let handler = move |_: HyperRequest, mut response: HyperResponse| {
         response.headers_mut().set(ContentEncoding(vec![Encoding::Gzip]));
         let mut e = GzEncoder::new(Vec::new(), Compression::Default);
@@ -427,26 +431,27 @@ fn test_load_should_decode_the_response_
     };
     let (mut server, url) = make_server(handler);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
-    assert_eq!(*response.body.lock().unwrap(),
+    let internal_response = response.internal_response.unwrap();
+    assert!(internal_response.status.unwrap().is_success());
+    assert_eq!(*internal_response.body.lock().unwrap(),
                ResponseBody::Done(b"Yay!".to_vec()));
 }
 
 #[test]
 fn test_load_doesnt_send_request_body_on_any_redirect() {
     let post_handler = move |mut request: HyperRequest, response: HyperResponse| {
         assert_eq!(request.method, Method::Get);
         let data = read_response(&mut request);
@@ -465,17 +470,17 @@ fn test_load_doesnt_send_request_body_on
     };
     let (mut pre_server, pre_url) = make_server(pre_handler);
 
     let mut request = Request::from_init(RequestInit {
         url: pre_url.clone(),
         body: Some(b"Body on POST!".to_vec()),
         method: Method::Post,
         destination: Destination::Document,
-        origin: pre_url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = pre_server.close();
     let _ = post_server.close();
 
@@ -490,26 +495,26 @@ fn test_load_doesnt_add_host_to_sts_list
     };
     let (mut server, url) = make_server(handler);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let context = new_fetch_context(None);
     let response = fetch_with_context(&mut request, &context);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
     assert_eq!(context.state.hsts_list.read().unwrap().is_host_secure(url.host_str().unwrap()), false);
 }
 
 #[test]
 fn test_load_sets_cookies_in_the_resource_manager_when_it_get_set_cookie_header_in_response() {
     let handler = move |_: HyperRequest, mut response: HyperResponse| {
         response.headers_mut().set(SetCookie(vec!["mozillaIs=theBest".to_owned()]));
         response.send(b"Yay!").unwrap();
@@ -520,26 +525,26 @@ fn test_load_sets_cookies_in_the_resourc
 
     assert_cookie_for_domain(&context.state.cookie_jar, url.as_str(), None);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         credentials_mode: CredentialsMode::Include,
         .. RequestInit::default()
     });
     let response = fetch_with_context(&mut request, &context);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 
     assert_cookie_for_domain(&context.state.cookie_jar, url.as_str(), Some("mozillaIs=theBest"));
 }
 
 #[test]
 fn test_load_sets_requests_cookies_header_for_url_by_getting_cookies_from_the_resource_manager() {
     let handler = move |request: HyperRequest, response: HyperResponse| {
         assert_eq!(request.headers.get::<CookieHeader>(),
@@ -560,26 +565,26 @@ fn test_load_sets_requests_cookies_heade
         cookie_jar.push(cookie, &url, CookieSource::HTTP);
     }
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         credentials_mode: CredentialsMode::Include,
         .. RequestInit::default()
     });
     let response = fetch_with_context(&mut request, &context);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 }
 
 #[test]
 fn test_load_sends_cookie_if_nonhttp() {
     let handler = move |request: HyperRequest, response: HyperResponse| {
         assert_eq!(request.headers.get::<CookieHeader>(),
                    Some(&CookieHeader(vec!["mozillaIs=theBest".to_owned()])));
         response.send(b"Yay!").unwrap();
@@ -598,26 +603,26 @@ fn test_load_sends_cookie_if_nonhttp() {
         cookie_jar.push(cookie, &url, CookieSource::HTTP);
     }
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         credentials_mode: CredentialsMode::Include,
         .. RequestInit::default()
     });
     let response = fetch_with_context(&mut request, &context);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 }
 
 #[test]
 fn test_cookie_set_with_httponly_should_not_be_available_using_getcookiesforurl() {
     let handler = move |_: HyperRequest, mut response: HyperResponse| {
         let pair = vec!["mozillaIs=theBest; HttpOnly".to_owned()];
         response.headers_mut().set(SetCookie(pair));
         response.send(b"Yay!").unwrap();
@@ -628,26 +633,26 @@ fn test_cookie_set_with_httponly_should_
 
     assert_cookie_for_domain(&context.state.cookie_jar, url.as_str(), None);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         credentials_mode: CredentialsMode::Include,
         .. RequestInit::default()
     });
     let response = fetch_with_context(&mut request, &context);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 
     assert_cookie_for_domain(&context.state.cookie_jar, url.as_str(), Some("mozillaIs=theBest"));
     let mut cookie_jar = context.state.cookie_jar.write().unwrap();
     assert!(cookie_jar.cookies_for_url(&url, CookieSource::NonHTTP).is_none());
 }
 
 #[test]
 fn test_when_cookie_received_marked_secure_is_ignored_for_http() {
@@ -662,26 +667,26 @@ fn test_when_cookie_received_marked_secu
 
     assert_cookie_for_domain(&context.state.cookie_jar, url.as_str(), None);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         credentials_mode: CredentialsMode::Include,
         .. RequestInit::default()
     });
     let response = fetch_with_context(&mut request, &context);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 
     assert_cookie_for_domain(&context.state.cookie_jar, url.as_str(), None);
 }
 
 #[test]
 fn test_load_sets_content_length_to_length_of_request_body() {
     let content = b"This is a request body";
     let content_length = ContentLength(content.len() as u64);
@@ -691,25 +696,25 @@ fn test_load_sets_content_length_to_leng
     };
     let (mut server, url) = make_server(handler);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Post,
         body: Some(content.to_vec()),
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 }
 
 #[test]
 fn test_load_uses_explicit_accept_from_headers_in_load_data() {
     let accept = Accept(vec![qitem(Mime(TopLevel::Text, SubLevel::Html, vec![]))]);
     let expected_accept = accept.clone();
     let handler = move |request: HyperRequest, response: HyperResponse| {
         assert_eq!(request.headers.get::<Accept>(), Some(&expected_accept));
@@ -719,25 +724,25 @@ fn test_load_uses_explicit_accept_from_h
 
     let mut accept_headers = Headers::new();
     accept_headers.set(accept);
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         headers: accept_headers,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 }
 
 #[test]
 fn test_load_sets_default_accept_to_html_xhtml_xml_and_then_anything_else() {
     let handler = move |request: HyperRequest, response: HyperResponse| {
         assert_eq!(request.headers.get::<Accept>(), Some(&Accept(vec![
             qitem(Mime(TopLevel::Text, SubLevel::Html, vec![])),
             qitem(Mime(TopLevel::Application, SubLevel::Ext("xhtml+xml".to_owned()), vec![])),
@@ -747,25 +752,25 @@ fn test_load_sets_default_accept_to_html
         response.send(b"Yay!").unwrap();
     };
     let (mut server, url) = make_server(handler);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 }
 
 #[test]
 fn test_load_uses_explicit_accept_encoding_from_load_data_headers() {
     let accept_encoding = AcceptEncoding(vec![qitem(Encoding::Chunked)]);
     let expected_accept_encoding = accept_encoding.clone();
     let handler = move |request: HyperRequest, response: HyperResponse| {
         assert_eq!(request.headers.get::<AcceptEncoding>(), Some(&expected_accept_encoding));
@@ -775,25 +780,25 @@ fn test_load_uses_explicit_accept_encodi
 
     let mut accept_encoding_headers = Headers::new();
     accept_encoding_headers.set(accept_encoding);
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         headers: accept_encoding_headers,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 }
 
 #[test]
 fn test_load_sets_default_accept_encoding_to_gzip_and_deflate() {
     let handler = move |request: HyperRequest, response: HyperResponse| {
         assert_eq!(request.headers.get::<AcceptEncoding>(), Some(&AcceptEncoding(vec![
             qitem(Encoding::Gzip),
             qitem(Encoding::Deflate),
@@ -802,25 +807,25 @@ fn test_load_sets_default_accept_encodin
         response.send(b"Yay!").unwrap();
     };
     let (mut server, url) = make_server(handler);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 }
 
 #[test]
 fn test_load_errors_when_there_a_redirect_loop() {
     let url_b_for_a = Arc::new(Mutex::new(None::<ServoUrl>));
     let url_b_for_a_clone = url_b_for_a.clone();
     let handler_a = move |_: HyperRequest, mut response: HyperResponse| {
         response.headers_mut().set(Location(url_b_for_a_clone.lock().unwrap().as_ref().unwrap().to_string()));
@@ -838,17 +843,17 @@ fn test_load_errors_when_there_a_redirec
     let (mut server_b, url_b) = make_server(handler_b);
 
     *url_b_for_a.lock().unwrap() = Some(url_b.clone());
 
     let mut request = Request::from_init(RequestInit {
         url: url_a.clone(),
         method: Method::Get,
         destination: Destination::Document,
-        origin: url_a.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = server_a.close();
     let _ = server_b.close();
 
@@ -881,17 +886,17 @@ fn test_load_succeeds_with_a_redirect_lo
     let (mut server_b, url_b) = make_server(handler_b);
 
     *url_b_for_a.lock().unwrap() = Some(url_b.clone());
 
     let mut request = Request::from_init(RequestInit {
         url: url_a.clone(),
         method: Method::Get,
         destination: Destination::Document,
-        origin: url_a.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = server_a.close();
     let _ = server_b.close();
 
@@ -917,28 +922,28 @@ fn test_load_follows_a_redirect() {
         response.send(b"").unwrap();
     };
     let (mut pre_server, pre_url) = make_server(pre_handler);
 
     let mut request = Request::from_init(RequestInit {
         url: pre_url.clone(),
         method: Method::Get,
         destination: Destination::Document,
-        origin: pre_url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = pre_server.close();
     let _ = post_server.close();
 
-    let response = response.to_actual();
-    assert!(response.status.unwrap().is_success());
-    assert_eq!(*response.body.lock().unwrap(),
+    let internal_response = response.internal_response.unwrap();
+    assert!(internal_response.status.unwrap().is_success());
+    assert_eq!(*internal_response.body.lock().unwrap(),
                ResponseBody::Done(b"Yay!".to_vec()));
 }
 
 #[test]
 fn  test_redirect_from_x_to_y_provides_y_cookies_from_y() {
     let shared_url_y = Arc::new(Mutex::new(None::<ServoUrl>));
     let shared_url_y_clone = shared_url_y.clone();
     let handler = move |request: HyperRequest, mut response: HyperResponse| {
@@ -994,28 +999,28 @@ fn  test_redirect_from_x_to_y_provides_y
         ).unwrap();
         cookie_jar.push(cookie_y, &url_y, CookieSource::HTTP);
     }
 
     let mut request = Request::from_init(RequestInit {
         url: url_x.clone(),
         method: Method::Get,
         destination: Destination::Document,
-        origin: url_x.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         credentials_mode: CredentialsMode::Include,
         .. RequestInit::default()
     });
     let response = fetch_with_context(&mut request, &context);
 
     let _ = server.close();
 
-    let response = response.to_actual();
-    assert!(response.status.unwrap().is_success());
-    assert_eq!(*response.body.lock().unwrap(),
+    let internal_response = response.internal_response.unwrap();
+    assert!(internal_response.status.unwrap().is_success());
+    assert_eq!(*internal_response.body.lock().unwrap(),
                ResponseBody::Done(b"Yay!".to_vec()));
 }
 
 #[test]
 fn test_redirect_from_x_to_x_provides_x_with_cookie_from_first_response() {
     let handler = move |request: HyperRequest, mut response: HyperResponse| {
         let path = match request.uri {
             ::hyper::uri::RequestUri::AbsolutePath(path) => path,
@@ -1038,28 +1043,28 @@ fn test_redirect_from_x_to_x_provides_x_
     let (mut server, url) = make_server(handler);
 
     let url = url.join("/initial/").unwrap();
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         credentials_mode: CredentialsMode::Include,
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
 
     let _ = server.close();
 
-    let response = response.to_actual();
-    assert!(response.status.unwrap().is_success());
-    assert_eq!(*response.body.lock().unwrap(),
+    let internal_response = response.internal_response.unwrap();
+    assert!(internal_response.status.unwrap().is_success());
+    assert_eq!(*internal_response.body.lock().unwrap(),
                ResponseBody::Done(b"Yay!".to_vec()));
 }
 
 #[test]
 fn test_if_auth_creds_not_in_url_but_in_cache_it_sets_it() {
     let handler = move |request: HyperRequest, response: HyperResponse| {
         let expected = Authorization(Basic {
             username: "username".to_owned(),
@@ -1070,17 +1075,17 @@ fn test_if_auth_creds_not_in_url_but_in_
     };
     let (mut server, url) = make_server(handler);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         credentials_mode: CredentialsMode::Include,
         .. RequestInit::default()
     });
     let context = new_fetch_context(None);
 
     let auth_entry = AuthCacheEntry {
         user_name: "username".to_owned(),
@@ -1088,43 +1093,43 @@ fn test_if_auth_creds_not_in_url_but_in_
     };
 
     context.state.auth_cache.write().unwrap().entries.insert(url.origin().clone().ascii_serialization(), auth_entry);
 
     let response = fetch_with_context(&mut request, &context);
 
     let _ = server.close();
 
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 }
 
 #[test]
 fn test_auth_ui_needs_www_auth() {
     let handler = move |_: HyperRequest, mut response: HyperResponse| {
         *response.status_mut() = StatusCode::Unauthorized;
         response.send(b"").unwrap();
     };
     let (mut server, url) = make_server(handler);
 
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         body: None,
         destination: Destination::Document,
-        origin: url.clone(),
+        origin: mock_origin(),
         pipeline_id: Some(TEST_PIPELINE_ID),
         credentials_mode: CredentialsMode::Include,
         .. RequestInit::default()
     });
 
     let response = fetch(&mut request, None);
 
     let _ = server.close();
 
-    assert_eq!(response.status.unwrap(), StatusCode::Unauthorized);
+    assert_eq!(response.internal_response.unwrap().status.unwrap(), StatusCode::Unauthorized);
 }
 
 #[test]
 fn test_origin_set() {
     let origin_header = Arc::new(Mutex::new(None));
     let origin_header_clone = origin_header.clone();
     let handler = move |request: HyperRequest, mut resp: HyperResponse| {
         let origin_header_clone = origin_header.clone();
@@ -1137,45 +1142,45 @@ fn test_origin_set() {
     let (mut server, url) = make_server(handler);
 
     let mut origin = Origin::new(url.scheme(), url.host_str().unwrap(), url.port());
     *origin_header_clone.lock().unwrap() = Some(origin.clone());
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Post,
         body: None,
-        origin: url.clone(),
+        origin: url.clone().origin(),
         .. RequestInit::default()
     });
     let response = fetch(&mut request, None);
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 
     let origin_url = ServoUrl::parse("http://example.com").unwrap();
     origin = Origin::new(origin_url.scheme(), origin_url.host_str().unwrap(), origin_url.port());
     // Test Origin header is set on Get request with CORS mode
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Get,
         mode: RequestMode::CorsMode,
         body: None,
-        origin: origin_url.clone(),
+        origin: origin_url.clone().origin(),
         .. RequestInit::default()
     });
 
     *origin_header_clone.lock().unwrap() = Some(origin.clone());
     let response = fetch(&mut request, None);
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 
     // Test Origin header is not set on method Head
     let mut request = Request::from_init(RequestInit {
         url: url.clone(),
         method: Method::Head,
         body: None,
-        origin: url.clone(),
+        origin: url.clone().origin(),
         .. RequestInit::default()
     });
 
     *origin_header_clone.lock().unwrap() = None;
     let response = fetch(&mut request, None);
-    assert!(response.status.unwrap().is_success());
+    assert!(response.internal_response.unwrap().status.unwrap().is_success());
 
     let _ = server.close();
 }
--- a/servo/tests/unit/style/animated_properties.rs
+++ b/servo/tests/unit/style/animated_properties.rs
@@ -3,17 +3,17 @@
  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 
 use app_units::Au;
 use cssparser::RGBA;
 use style::properties::animated_properties::{Animatable, IntermediateRGBA};
 use style::properties::longhands::transform::computed_value::ComputedOperation as TransformOperation;
 use style::properties::longhands::transform::computed_value::T as TransformList;
 use style::values::animated::ToAnimatedValue;
-use style::values::specified::length::Percentage;
+use style::values::computed::length::Percentage;
 
 fn interpolate_rgba(from: RGBA, to: RGBA, progress: f64) -> RGBA {
     let from = from.to_animated_value();
     let to = to.to_animated_value();
     RGBA::from_animated_value(from.interpolate(&to, progress).unwrap())
 }
 
 // Color
--- a/servo/tests/unit/style/attr.rs
+++ b/servo/tests/unit/style/attr.rs
@@ -1,16 +1,16 @@
 /* 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/. */
 
 use app_units::Au;
 use style::attr::{AttrValue, LengthOrPercentageOrAuto, parse_length};
 use style::values::computed::CalcLengthOrPercentage;
-use style::values::specified::length::Percentage;
+use style::values::computed::length::Percentage;
 
 #[test]
 fn test_length_calc() {
     let calc = CalcLengthOrPercentage::new(Au(10), Some(Percentage(0.2)));
     assert_eq!(calc.to_used_value(Some(Au(10))), Some(Au(12)));
     assert_eq!(calc.to_used_value(Some(Au(0))), Some(Au(10)));
     assert_eq!(calc.to_used_value(None), None);
 
--- a/servo/tests/unit/style/properties/serialization.rs
+++ b/servo/tests/unit/style/properties/serialization.rs
@@ -353,26 +353,26 @@ mod shorthand_serialization {
 
         use style::values::specified::BorderCornerRadius;
         use style::values::specified::length::Percentage;
 
         #[test]
         fn border_radius_should_serialize_correctly() {
             let mut properties = Vec::new();
             properties.push(PropertyDeclaration::BorderTopLeftRadius(Box::new(BorderCornerRadius::new(
-                Percentage(0.01).into(), Percentage(0.05).into()
+                Percentage::new(0.01).into(), Percentage::new(0.05).into()
             ))));
             properties.push(PropertyDeclaration::BorderTopRightRadius(Box::new(BorderCornerRadius::new(
-                Percentage(0.02).into(), Percentage(0.06).into()
+                Percentage::new(0.02).into(), Percentage::new(0.06).into()
             ))));
             properties.push(PropertyDeclaration::BorderBottomRightRadius(Box::new(BorderCornerRadius::new(
-                Percentage(0.03).into(), Percentage(0.07).into()
+                Percentage::new(0.03).into(), Percentage::new(0.07).into()
             ))));
             properties.push(PropertyDeclaration::BorderBottomLeftRadius(Box::new(BorderCornerRadius::new(
-                Percentage(0.04).into(), Percentage(0.08).into()
+                Percentage::new(0.04).into(), Percentage::new(0.08).into()
             ))));
 
             let serialization = shorthand_properties_to_string(properties);
             assert_eq!(serialization, "border-radius: 1% 2% 3% 4% / 5% 6% 7% 8%;");
         }
     }
 
 
@@ -558,17 +558,17 @@ mod shorthand_serialization {
     fn flex_should_serialize_all_available_properties() {
         use style::values::specified::{Number, Percentage};
 
         let mut properties = Vec::new();
 
         let grow = Number::new(2f32);
         let shrink = Number::new(3f32);
         let basis =
-            FlexBasis::Length(Percentage(0.5f32).into());
+            FlexBasis::Length(Percentage::new(0.5f32).into());
 
         properties.push(PropertyDeclaration::FlexGrow(grow));
         properties.push(PropertyDeclaration::FlexShrink(shrink));
         properties.push(PropertyDeclaration::FlexBasis(basis));
 
         let serialization = shorthand_properties_to_string(properties);
         assert_eq!(serialization, "flex: 2 3 50%;");
     }
--- a/servo/tests/unit/style/stylesheets.rs
+++ b/servo/tests/unit/style/stylesheets.rs
@@ -21,17 +21,18 @@ use style::properties::{CSSWideKeyword, 
 use style::properties::longhands;
 use style::properties::longhands::animation_play_state;
 use style::shared_lock::SharedRwLock;
 use style::stylearc::Arc;
 use style::stylesheets::{Origin, Namespaces};
 use style::stylesheets::{Stylesheet, StylesheetContents, NamespaceRule, CssRule, CssRules, StyleRule, KeyframesRule};
 use style::stylesheets::keyframes_rule::{Keyframe, KeyframeSelector, KeyframePercentage};
 use style::values::{KeyframesName, CustomIdent};
-use style::values::specified::{LengthOrPercentageOrAuto, Percentage, PositionComponent};
+use style::values::computed::Percentage;
+use style::values::specified::{LengthOrPercentageOrAuto, PositionComponent};
 
 pub fn block_from<I>(iterable: I) -> PropertyDeclarationBlock
 where I: IntoIterator<Item=(PropertyDeclaration, Importance)> {
     let mut block = PropertyDeclarationBlock::new();
     for (d, i) in iterable {
         block.push(d, i)
     }
     block
@@ -310,21 +311,48 @@ fn test_report_error_stylesheet() {
     let media = Arc::new(lock.wrap(MediaList::empty()));
     Stylesheet::from_str(css, url.clone(), Origin::UserAgent, media, lock,
                          None, &error_reporter, QuirksMode::NoQuirks, 5u64);
 
     let mut errors = errors.lock().unwrap();
 
     let error = errors.pop().unwrap();
     assert_eq!("Unsupported property declaration: 'invalid: true;', \
-                Custom(UnknownProperty(\"invalid\"))", error.message);
+                Custom(PropertyDeclaration(UnknownProperty(\"invalid\")))", error.message);
     assert_eq!(9, error.line);
     assert_eq!(8, error.column);
 
     let error = errors.pop().unwrap();
     assert_eq!("Unsupported property declaration: 'display: invalid;', \
                 Custom(PropertyDeclaration(InvalidValue(\"display\")))", error.message);
     assert_eq!(8, error.line);
     assert_eq!(8, error.column);
 
     // testing for the url
     assert_eq!(url, error.url);
 }
+
+#[test]
+fn test_no_report_unrecognized_vendor_properties() {
+    let css = r"
+    div {
+        -o-background-color: red;
+        _background-color: red;
+        -moz-background-color: red;
+    }
+    ";
+    let url = ServoUrl::parse("about::test").unwrap();
+    let error_reporter = CSSInvalidErrorReporterTest::new();
+
+    let errors = error_reporter.errors.clone();
+
+    let lock = SharedRwLock::new();
+    let media = Arc::new(lock.wrap(MediaList::empty()));
+    Stylesheet::from_str(css, url, Origin::UserAgent, media, lock,
+                         None, &error_reporter, QuirksMode::NoQuirks, 0u64);
+
+    let mut errors = errors.lock().unwrap();
+    let error = errors.pop().unwrap();
+    assert_eq!("Unsupported property declaration: '-moz-background-color: red;', \
+                Custom(PropertyDeclaration(UnknownProperty(\"-moz-background-color\")))",
+               error.message);
+    assert!(errors.is_empty());
+}
--- a/taskcluster/ci/source-test/mozlint.yml
+++ b/taskcluster/ci/source-test/mozlint.yml
@@ -8,19 +8,17 @@ mozlint-eslint:
     worker-type: aws-provisioner-v1/gecko-t-linux-xlarge
     worker:
         docker-image: {in-tree: "lint"}
         max-run-time: 1800
     run:
         using: run-task
         command: >
             cd /home/worker/checkouts/gecko/ &&
-            /build/tooltool.py fetch -m tools/lint/eslint/manifest.tt &&
-            tar xvfz eslint.tar.gz &&
-            rm eslint.tar.gz &&
+            cp -r /build/node_modules_eslint node_modules &&
             ln -s ../tools/lint/eslint/eslint-plugin-mozilla node_modules &&
             ln -s ../tools/lint/eslint/eslint-plugin-spidermonkey-js node_modules &&
             ./mach lint -l eslint -f treeherder --quiet
     when:
         files-changed:
             # Files that are likely audited.
             - '**/*.js'
             - '**/*.jsm'
--- a/taskcluster/docker/lint/Dockerfile
+++ b/taskcluster/docker/lint/Dockerfile
@@ -12,16 +12,18 @@ ADD topsrcdir/python/mozbuild/mozbuild/a
 ADD topsrcdir/testing/mozharness/external_tools/robustcheckout.py /usr/local/mercurial/robustcheckout.py
 
 # %include taskcluster/docker/recipes/install-node.sh
 ADD topsrcdir/taskcluster/docker/recipes/install-node.sh /build/install-node.sh
 
 # %include taskcluster/docker/recipes/install-mercurial.sh
 ADD topsrcdir/taskcluster/docker/recipes/install-mercurial.sh /build/install-mercurial.sh
 ADD system-setup.sh /tmp/system-setup.sh
+# %include tools/lint/eslint/manifest.tt
+ADD topsrcdir/tools/lint/eslint/manifest.tt /tmp/eslint.tt
 # %include tools/lint/eslint/eslint-plugin-mozilla/manifest.tt
 ADD topsrcdir/tools/lint/eslint/eslint-plugin-mozilla/manifest.tt /tmp/eslint-plugin-mozilla.tt
 # %include tools/lint/flake8_/flake8_requirements.txt
 ADD topsrcdir/tools/lint/flake8_/flake8_requirements.txt /tmp/flake8_requirements.txt
 # %include tools/lint/tox/tox_requirements.txt
 ADD topsrcdir/tools/lint/tox/tox_requirements.txt /tmp/tox_requirements.txt
 RUN bash /tmp/system-setup.sh
 
--- a/taskcluster/docker/lint/system-setup.sh
+++ b/taskcluster/docker/lint/system-setup.sh
@@ -42,16 +42,18 @@ cd /build
 ###
 # ESLint Setup
 ###
 
 # install node
 
 . install-node.sh
 
+/build/tooltool.py fetch -m /tmp/eslint.tt
+mv /build/node_modules /build/node_modules_eslint
 /build/tooltool.py fetch -m /tmp/eslint-plugin-mozilla.tt
 mv /build/node_modules /build/node_modules_eslint-plugin-mozilla
 
 ###
 # Flake8 Setup
 ###
 
 cd /setup
--- a/testing/marionette/harness/marionette_harness/tests/unit/test_navigation.py
+++ b/testing/marionette/harness/marionette_harness/tests/unit/test_navigation.py
@@ -379,17 +379,16 @@ class TestBackForwardNavigation(BaseNavi
     def test_file_url(self):
         test_pages = [
             {"url": self.test_page_remote},
             {"url": self.test_page_file_url},
             {"url": self.test_page_remote},
         ]
         self.run_bfcache_test(test_pages)
 
-    @skip("Causes crashes for JS GC (bug 1344863) and a11y (bug 1344868)")
     def test_frameset(self):
         test_pages = [
             {"url": self.marionette.absolute_url("frameset.html")},
             {"url": self.test_page_remote},
             {"url": self.marionette.absolute_url("frameset.html")},
         ]
         self.run_bfcache_test(test_pages)
 
--- a/toolkit/components/extensions/ExtensionTabs.jsm
+++ b/toolkit/components/extensions/ExtensionTabs.jsm
@@ -261,16 +261,27 @@ class TabBase {
    */
   get favIconUrl() {
     if (this.hasTabPermission) {
       return this._favIconUrl;
     }
   }
 
   /**
+   * @property {integer} lastAccessed
+   *        Returns the last time the tab was accessed as the number of
+   *        milliseconds since epoch.
+   *        @readonly
+   *        @abstract
+   */
+  get lastAccessed() {
+    throw new Error("Not implemented");
+  }
+
+  /**
    * @property {boolean} audible
    *        Returns true if the tab is currently playing audio, false otherwise.
    *        @readonly
    *        @abstract
    */
   get audible() {
     throw new Error("Not implemented");
   }
@@ -476,16 +487,17 @@ class TabBase {
       windowId: this.windowId,
       highlighted: this.selected,
       active: this.selected,
       pinned: this.pinned,
       status: this.status,
       incognito: this.incognito,
       width: this.width,
       height: this.height,
+      lastAccessed: this.lastAccessed,
       audible: this.audible,
       mutedInfo: this.mutedInfo,
     };
 
     // If the tab has not been fully layed-out yet, fallback to the geometry
     // from a different tab (usually the currently active tab).
     if (fallbackTab && (!result.width || !result.height)) {
       result.width = fallbackTab.width;
--- a/toolkit/components/extensions/test/mochitest/mochitest-common.ini
+++ b/toolkit/components/extensions/test/mochitest/mochitest-common.ini
@@ -122,9 +122,9 @@ skip-if = os == 'android'
 [test_ext_webrequest_upload.html]
 skip-if = os == 'android' # Currently fails in emulator tests
 [test_ext_webrequest_permission.html]
 [test_ext_webrequest_websocket.html]
 [test_ext_webnavigation.html]
 [test_ext_webnavigation_filters.html]
 [test_ext_window_postMessage.html]
 [test_ext_subframes_privileges.html]
-[test_ext_xhr_capabilities.html]
\ No newline at end of file
+[test_ext_xhr_capabilities.html]
--- a/toolkit/components/places/Bookmarks.jsm
+++ b/toolkit/components/places/Bookmarks.jsm
@@ -2181,17 +2181,19 @@ async function(db, folderGuids, options)
          JOIN descendants ON parent = did
        )
        SELECT b.id AS _id, b.parent AS _parentId, b.position AS 'index',
               b.type, url, b.guid, p.guid AS parentGuid, b.dateAdded,
               b.lastModified, IFNULL(b.title, "") AS title,
               p.parent AS _grandParentId, NULL AS _childCount,
               b.syncStatus AS _syncStatus
        FROM descendants
-       JOIN moz_bookmarks b ON did = b.id
+       /* The usage of CROSS JOIN is not random, it tells the optimizer
+          to retain the original rows order, so the hierarchy is respected */
+       CROSS JOIN moz_bookmarks b ON did = b.id
        JOIN moz_bookmarks p ON p.id = b.parent
        LEFT JOIN moz_places h ON b.fk = h.id`, { folderGuid });
 
     itemsRemoved = itemsRemoved.concat(rowsToItemsArray(rows));
 
     await db.executeCached(
       `WITH RECURSIVE
        descendants(did) AS (
--- a/toolkit/components/places/Database.cpp
+++ b/toolkit/components/places/Database.cpp
@@ -153,62 +153,16 @@ hasRecentCorruptDB()
       NS_ENSURE_TRUE(lastMod > 0, false);
       return (PR_Now() - lastMod) > RECENT_BACKUP_TIME_MICROSEC;
     }
   }
   return false;
 }
 
 /**
- * Updates sqlite_stat1 table through ANALYZE.
- * Since also nsPlacesExpiration.js executes ANALYZE, the analyzed tables
- * must be the same in both components.  So ensure they are in sync.
- *
- * @param aDBConn
- *        The database connection.
- */
-nsresult
-updateSQLiteStatistics(mozIStorageConnection* aDBConn)
-{
-  MOZ_ASSERT(NS_IsMainThread());
-  nsCOMPtr<mozIStorageAsyncStatement> analyzePlacesStmt;
-  aDBConn->CreateAsyncStatement(NS_LITERAL_CSTRING(
-    "ANALYZE moz_places"
-  ), getter_AddRefs(analyzePlacesStmt));
-  NS_ENSURE_STATE(analyzePlacesStmt);
-  nsCOMPtr<mozIStorageAsyncStatement> analyzeBookmarksStmt;
-  aDBConn->CreateAsyncStatement(NS_LITERAL_CSTRING(
-    "ANALYZE moz_bookmarks"
-  ), getter_AddRefs(analyzeBookmarksStmt));
-  NS_ENSURE_STATE(analyzeBookmarksStmt);
-  nsCOMPtr<mozIStorageAsyncStatement> analyzeVisitsStmt;
-  aDBConn->CreateAsyncStatement(NS_LITERAL_CSTRING(
-    "ANALYZE moz_historyvisits"
-  ), getter_AddRefs(analyzeVisitsStmt));
-  NS_ENSURE_STATE(analyzeVisitsStmt);
-  nsCOMPtr<mozIStorageAsyncStatement> analyzeInputStmt;
-  aDBConn->CreateAsyncStatement(NS_LITERAL_CSTRING(
-    "ANALYZE moz_inputhistory"
-  ), getter_AddRefs(analyzeInputStmt));
-  NS_ENSURE_STATE(analyzeInputStmt);
-
-  mozIStorageBaseStatement *stmts[] = {
-    analyzePlacesStmt,
-    analyzeBookmarksStmt,
-    analyzeVisitsStmt,
-    analyzeInputStmt
-  };
-
-  nsCOMPtr<mozIStoragePendingStatement> ps;
-  (void)aDBConn->ExecuteAsync(stmts, ArrayLength(stmts), nullptr,
-                              getter_AddRefs(ps));
-  return NS_OK;
-}
-
-/**
  * Sets the connection journal mode to one of the JOURNAL_* types.
  *
  * @param aDBConn
  *        The database connection.
  * @param aJournalMode
  *        One of the JOURNAL_* types.
  * @returns the current journal mode.
  * @note this may return a different journal mode than the required one, since
@@ -639,21 +593,16 @@ Database::EnsureConnection()
       // Bail out if we couldn't fix the database.
       NS_ENSURE_SUCCESS(rv, rv);
     }
 
     if (databaseMigrated) {
       mDatabaseStatus = nsINavHistoryService::DATABASE_STATUS_UPGRADED;
     }
 
-    if (mDatabaseStatus == nsINavHistoryService::DATABASE_STATUS_UPGRADED ||
-        mDatabaseStatus == nsINavHistoryService::DATABASE_STATUS_CREATE) {
-      MOZ_ALWAYS_SUCCEEDS(updateSQLiteStatistics(mMainConn));
-    }
-
     // Initialize here all the items that are not part of the on-disk database,
     // like views, temp triggers or temp tables.  The database should not be
     // considered corrupt if any of the following fails.
 
     rv = InitTempEntities();
     NS_ENSURE_SUCCESS(rv, rv);
 
     initSucceeded = true;
@@ -2621,16 +2570,23 @@ Database::Shutdown()
     new FinalizeStatementCacheProxy<mozIStorageStatement>(
           mAsyncThreadStatements,
           NS_ISUPPORTS_CAST(nsIObserver*, this)
         );
   DispatchToAsyncThread(event);
 
   mClosed = true;
 
+  // Execute PRAGMA optimized as last step, this will ensure proper database
+  // performance across restarts.
+  nsCOMPtr<mozIStoragePendingStatement> ps;
+  MOZ_ALWAYS_SUCCEEDS(mMainConn->ExecuteSimpleSQLAsync(NS_LITERAL_CSTRING(
+    "PRAGMA optimize(0x02)"
+  ), nullptr, getter_AddRefs(ps)));
+
   (void)mMainConn->AsyncClose(connectionShutdown);
   mMainConn = nullptr;
 }
 
 ////////////////////////////////////////////////////////////////////////////////
 //// nsIObserver
 
 NS_IMETHODIMP
--- a/toolkit/components/places/nsPlacesExpiration.js
+++ b/toolkit/components/places/nsPlacesExpiration.js
@@ -109,20 +109,16 @@ const URIENTRY_AVG_SIZE = 600;
 // stand-by or mobile devices batteries.
 const IDLE_TIMEOUT_SECONDS = 5 * 60;
 
 // If a clear history ran just before we shutdown, we will skip most of the
 // expiration at shutdown.  This is maximum number of seconds from last
 // clearHistory to decide to skip expiration at shutdown.
 const SHUTDOWN_WITH_RECENT_CLEARHISTORY_TIMEOUT_SECONDS = 10;
 
-// If the pages delta from the last ANALYZE is over this threashold, the tables
-// should be analyzed again.
-const ANALYZE_PAGES_THRESHOLD = 100;
-
 // If the number of pages over history limit is greater than this threshold,
 // expiration will be more aggressive, to bring back history to a saner size.
 const OVERLIMIT_PAGES_THRESHOLD = 1000;
 
 const MSECS_PER_DAY = 86400000;
 const ANNOS_EXPIRE_POLICIES = [
   { bind: "expire_days",
     type: Ci.nsIAnnotationService.EXPIRE_DAYS,
@@ -153,22 +149,21 @@ const STATUS = {
   DIRTY: 1,
   UNKNOWN: 2,
 };
 
 // Represents actions on which a query will run.
 const ACTION = {
   TIMED:           1 << 0, // happens every this._interval
   TIMED_OVERLIMIT: 1 << 1, // like TIMED but only when history is over limits
-  TIMED_ANALYZE:   1 << 2, // happens when ANALYZE statistics should be updated
-  CLEAR_HISTORY:   1 << 3, // happens when history is cleared
-  SHUTDOWN_DIRTY:  1 << 4, // happens at shutdown for DIRTY state
-  IDLE_DIRTY:      1 << 5, // happens on idle for DIRTY state
-  IDLE_DAILY:      1 << 6, // happens once a day on idle
-  DEBUG:           1 << 7, // happens on TOPIC_DEBUG_START_EXPIRATION
+  CLEAR_HISTORY:   1 << 2, // happens when history is cleared
+  SHUTDOWN_DIRTY:  1 << 3, // happens at shutdown for DIRTY state
+  IDLE_DIRTY:      1 << 4, // happens on idle for DIRTY state
+  IDLE_DAILY:      1 << 5, // happens once a day on idle
+  DEBUG:           1 << 6, // happens on TOPIC_DEBUG_START_EXPIRATION
 };
 
 // The queries we use to expire.
 const EXPIRATION_QUERIES = {
 
   // Some visits can be expired more often than others, cause they are less
   // useful to the user and can pollute awesomebar results:
   // 1. urls over 255 chars
@@ -409,42 +404,16 @@ const EXPIRATION_QUERIES = {
   },
 
   // Empty the notifications table.
   QUERY_DELETE_NOTIFICATIONS: {
     sql: "DELETE FROM expiration_notify",
     actions: ACTION.TIMED | ACTION.TIMED_OVERLIMIT | ACTION.SHUTDOWN_DIRTY |
              ACTION.IDLE_DIRTY | ACTION.IDLE_DAILY | ACTION.DEBUG
   },
-
-  // The following queries are used to adjust the sqlite_stat1 table to help the
-  // query planner create better queries.  These should always be run LAST, and
-  // are therefore at the end of the object.
-  // Since also nsNavHistory.cpp executes ANALYZE, the analyzed tables
-  // must be the same in both components.  So ensure they are in sync.
-
-  QUERY_ANALYZE_MOZ_PLACES: {
-    sql: "ANALYZE moz_places",
-    actions: ACTION.TIMED_OVERLIMIT | ACTION.TIMED_ANALYZE |
-             ACTION.CLEAR_HISTORY | ACTION.IDLE_DAILY | ACTION.DEBUG
-  },
-  QUERY_ANALYZE_MOZ_BOOKMARKS: {
-    sql: "ANALYZE moz_bookmarks",
-    actions: ACTION.TIMED_ANALYZE | ACTION.IDLE_DAILY | ACTION.DEBUG
-  },
-  QUERY_ANALYZE_MOZ_HISTORYVISITS: {
-    sql: "ANALYZE moz_historyvisits",
-    actions: ACTION.TIMED_OVERLIMIT | ACTION.TIMED_ANALYZE |
-             ACTION.CLEAR_HISTORY | ACTION.IDLE_DAILY | ACTION.DEBUG
-  },
-  QUERY_ANALYZE_MOZ_INPUTHISTORY: {
-    sql: "ANALYZE moz_inputhistory",
-    actions: ACTION.TIMED | ACTION.TIMED_OVERLIMIT | ACTION.TIMED_ANALYZE |
-             ACTION.CLEAR_HISTORY | ACTION.IDLE_DAILY | ACTION.DEBUG
-  },
 };
 
 /**
  * Sends a bookmarks notification through the given observers.
  *
  * @param observers
  *        array of nsINavBookmarkObserver objects.
  * @param notification
@@ -621,27 +590,20 @@ nsPlacesExpiration.prototype = {
   onDeleteURI() {},
   onPageChanged() {},
   onDeleteVisits() {},
 
   // nsITimerCallback
 
   notify: function PEX_timerCallback() {
     // Check if we are over history capacity, if so visits must be expired.
-    this._getPagesStats((aPagesCount, aStatsCount) => {
+    this._getPagesStats((aPagesCount) => {
       let overLimitPages = aPagesCount - this._urisLimit;
       this._overLimit = overLimitPages > 0;
-
       let action = this._overLimit ? ACTION.TIMED_OVERLIMIT : ACTION.TIMED;
-      // If the number of pages changed significantly from the last ANALYZE
-      // update SQLite statistics.
-      if (Math.abs(aPagesCount - aStatsCount) >= ANALYZE_PAGES_THRESHOLD) {
-        action = action | ACTION.TIMED_ANALYZE;
-      }
-
       // Adapt expiration aggressivity to the number of pages over the limit.
       let limit = overLimitPages > OVERLIMIT_PAGES_THRESHOLD ? LIMIT.LARGE
                                                              : LIMIT.SMALL;
 
       this._expireWithActionAndLimit(action, limit);
     });
   },
 
@@ -880,37 +842,33 @@ nsPlacesExpiration.prototype = {
     }
   },
 
   /**
    * Evaluates the real number of pages in the database and the value currently
    * used by the SQLite query planner.
    *
    * @param aCallback
-   *        invoked on success, function (aPagesCount, aStatsCount).
+   *        invoked on success, function (aPagesCount).
    */
   _getPagesStats: function PEX__getPagesStats(aCallback) {
     if (!this._cachedStatements["LIMIT_COUNT"]) {
       this._cachedStatements["LIMIT_COUNT"] = this._db.createAsyncStatement(
-        `SELECT (SELECT COUNT(*) FROM moz_places),
-                (SELECT SUBSTR(stat,1,LENGTH(stat)-2) FROM sqlite_stat1
-                 WHERE idx = 'moz_places_url_uniqueindex')`
+        `SELECT COUNT(*) FROM moz_places`
       );
     }
     this._cachedStatements["LIMIT_COUNT"].executeAsync({
       _pagesCount: 0,
-      _statsCount: 0,
       handleResult(aResults) {
         let row = aResults.getNextRow();
         this._pagesCount = row.getResultByIndex(0);
-        this._statsCount = row.getResultByIndex(1);
       },
       handleCompletion(aReason) {
         if (aReason == Ci.mozIStorageStatementCallback.REASON_FINISHED) {
-          aCallback(this._pagesCount, this._statsCount);
+          aCallback(this._pagesCount);
         }
       },
       handleError(aError) {
         Cu.reportError("Async statement execution returned with '" +
                        aError.result + "', '" + aError.message + "'");
       }
     });
   },
--- a/toolkit/components/places/tests/browser/browser_visituri_privatebrowsing_perwindowpb.js
+++ b/toolkit/components/places/tests/browser/browser_visituri_privatebrowsing_perwindowpb.js
@@ -1,73 +1,61 @@
 /* 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/. */
 
-function test() {
-  // initialization
-  waitForExplicitFinish();
-  let windowsToClose = [];
-  let initialURL =
-    "http://example.com/tests/toolkit/components/places/tests/browser/begin.html";
-  let finalURL =
-    "http://example.com/tests/toolkit/components/places/tests/browser/final.html";
-  let observer = null;
-  let enumerator = null;
-  let currentObserver = null;
-  let uri = null;
+const initialURL =
+  "http://example.com/tests/toolkit/components/places/tests/browser/begin.html";
+const finalURL =
+ "http://example.com/tests/toolkit/components/places/tests/browser/final.html";
+
+var observer;
+var visitSavedPromise;
 
-  function doTest(aIsPrivateMode, aWindow, aTestURI, aCallback) {
+add_task(async function setup() {
+  visitSavedPromise = new Promise(resolve => {
     observer = {
-      observe(aSubject, aTopic, aData) {
+      observe(subject, topic, data) {
         // The uri-visit-saved topic should only work when on normal mode.
-        if (aTopic == "uri-visit-saved") {
-          // Remove the observers set on per window private mode and normal
-          // mode.
-          enumerator = aWindow.Services.obs.enumerateObservers("uri-visit-saved");
-          while (enumerator.hasMoreElements()) {
-            currentObserver = enumerator.getNext();
-            aWindow.Services.obs.removeObserver(currentObserver, "uri-visit-saved");
-          }
+        if (topic == "uri-visit-saved") {
+          Services.obs.removeObserver(observer, "uri-visit-saved");
 
           // The expected visit should be the finalURL because private mode
           // should not register a visit with the initialURL.
-          uri = aSubject.QueryInterface(Ci.nsIURI);
-          is(uri.spec, finalURL, "Check received expected visit");
+          let uri = subject.QueryInterface(Ci.nsIURI);
+          resolve(uri.spec);
         }
       }
     };
-
-    aWindow.Services.obs.addObserver(observer, "uri-visit-saved");
-
-    BrowserTestUtils.browserLoaded(aWindow.gBrowser.selectedBrowser).then(aCallback);
-    aWindow.gBrowser.selectedBrowser.loadURI(aTestURI);
-  }
-
-  function testOnWindow(aOptions, aCallback) {
-    whenNewWindowLoaded(aOptions, function(aWin) {
-      windowsToClose.push(aWin);
-      // execute should only be called when need, like when you are opening
-      // web pages on the test. If calling executeSoon() is not necesary, then
-      // call whenNewWindowLoaded() instead of testOnWindow() on your test.
-      executeSoon(() => aCallback(aWin));
-    });
-  }
-
-   // This function is called after calling finish() on the test.
-  registerCleanupFunction(function() {
-    windowsToClose.forEach(function(aWin) {
-      aWin.close();
-    });
   });
 
-  // test first when on private mode
-  testOnWindow({private: true}, function(aWin) {
-    doTest(true, aWin, initialURL, function() {
-      // then test when not on private mode
-      testOnWindow({}, function(aWin2) {
-        doTest(false, aWin2, finalURL, function() {
-          PlacesTestUtils.clearHistory().then(finish);
-        });
-      });
-    });
+  Services.obs.addObserver(observer, "uri-visit-saved");
+
+  registerCleanupFunction(async function() {
+    await PlacesTestUtils.clearHistory()
   });
+});
+
+// Note: The private window test must be the first one to run, since we'll listen
+// to the first uri-visit-saved notification, and we expect this test to not
+// fire any, so we'll just find the non-private window test notification.
+add_task(async function test_private_browsing_window() {
+  await testLoadInWindow({private: true}, initialURL);
+});
+
+add_task(async function test_normal_window() {
+  await testLoadInWindow({private: false}, finalURL);
+
+  let url = await visitSavedPromise;
+  Assert.equal(url, finalURL, "Check received expected visit");
+});
+
+async function testLoadInWindow(options, url) {
+  let win = await BrowserTestUtils.openNewBrowserWindow(options);
+
+  registerCleanupFunction(async function() {
+    await BrowserTestUtils.closeWindow(win);
+  });
+
+  let loadedPromise = BrowserTestUtils.browserLoaded(win.gBrowser.selectedBrowser);
+  await BrowserTestUtils.loadURI(win.gBrowser.selectedBrowser, url);
+  await loadedPromise;
 }
deleted file mode 100644
--- a/toolkit/components/places/tests/expiration/test_analyze_runs.js
+++ /dev/null
@@ -1,111 +0,0 @@
-/* Any copyright is dedicated to the Public Domain.
-   http://creativecommons.org/publicdomain/zero/1.0/ */
-
-// Constants
-
-const TOPIC_AUTOCOMPLETE_FEEDBACK_INCOMING = "autocomplete-will-enter-text";
-
-// Helpers
-
-/**
- * Ensures that we have no data in the tables created by ANALYZE.
- */
-function clearAnalyzeData() {
-  let db = DBConn();
-  if (!db.tableExists("sqlite_stat1")) {
-    return;
-  }
-  db.executeSimpleSQL("DELETE FROM sqlite_stat1");
-}
-
-/**
- * Checks that we ran ANALYZE on the specified table.
- *
- * @param aTableName
- *        The table to check if ANALYZE was ran.
- * @param aRan
- *        True if it was expected to run, false otherwise
- */
-function do_check_analyze_ran(aTableName, aRan) {
-  let db = DBConn();
-  do_check_true(db.tableExists("sqlite_stat1"));
-  let stmt = db.createStatement("SELECT idx FROM sqlite_stat1 WHERE tbl = :table");
-  stmt.params.table = aTableName;
-  try {
-    if (aRan) {
-      do_check_true(stmt.executeStep());
-      do_check_neq(stmt.row.idx, null);
-    } else {
-      do_check_false(stmt.executeStep());
-    }
-  } finally {
-    stmt.finalize();
-  }
-}
-
-// Tests
-
-add_task(async function init_tests() {
-  const TEST_URI = NetUtil.newURI("http://mozilla.org/");
-  const TEST_TITLE = "This is a test";
-
-  await PlacesUtils.bookmarks.insert({
-    parentGuid: PlacesUtils.bookmarks.unfiledGuid,
-    title: TEST_TITLE,
-    url: TEST_URI
-  });
-  await PlacesTestUtils.addVisits(TEST_URI);
-  let thing = {
-    QueryInterface: XPCOMUtils.generateQI([Ci.nsIAutoCompleteInput,
-                                           Ci.nsIAutoCompletePopup,
-                                           Ci.nsIAutoCompleteController]),
-    get popup() { return thing; },
-    get controller() { return thing; },
-    popupOpen: true,
-    selectedIndex: 0,
-    getValueAt() { return TEST_URI.spec; },
-    searchString: TEST_TITLE,
-  };
-  Services.obs.notifyObservers(thing, TOPIC_AUTOCOMPLETE_FEEDBACK_INCOMING);
-});
-
-add_task(async function test_timed() {
-  clearAnalyzeData();
-
-  // Set a low interval and wait for the timed expiration to start.
-  let promise = promiseTopicObserved(PlacesUtils.TOPIC_EXPIRATION_FINISHED);
-  setInterval(3);
-  await promise;
-  setInterval(3600);
-
-  do_check_analyze_ran("moz_places", false);
-  do_check_analyze_ran("moz_bookmarks", false);
-  do_check_analyze_ran("moz_historyvisits", false);
-  do_check_analyze_ran("moz_inputhistory", true);
-});
-
-add_task(async function test_debug() {
-  clearAnalyzeData();
-
-  await promiseForceExpirationStep(1);
-
-  do_check_analyze_ran("moz_places", true);
-  do_check_analyze_ran("moz_bookmarks", true);
-  do_check_analyze_ran("moz_historyvisits", true);
-  do_check_analyze_ran("moz_inputhistory", true);
-});
-
-add_task(async function test_clear_history() {
-  clearAnalyzeData();
-
-  let promise = promiseTopicObserved(PlacesUtils.TOPIC_EXPIRATION_FINISHED);
-  let listener = Cc["@mozilla.org/places/expiration;1"]
-                 .getService(Ci.nsINavHistoryObserver);
-  listener.onClearHistory();
-  await promise;
-
-  do_check_analyze_ran("moz_places", true);
-  do_check_analyze_ran("moz_bookmarks", false);
-  do_check_analyze_ran("moz_historyvisits", true);
-  do_check_analyze_ran("moz_inputhistory", true);
-});
deleted file mode 100644
--- a/toolkit/components/places/tests/expiration/test_outdated_analyze.js
+++ /dev/null
@@ -1,72 +0,0 @@
-/* Any copyright is dedicated to the Public Domain.
-   http://creativecommons.org/publicdomain/zero/1.0/ */
-
-// Test that expiration executes ANALYZE when statistics are outdated.
-
-const TEST_URL = "http://www.mozilla.org/";
-
-XPCOMUtils.defineLazyServiceGetter(this, "gHistory",
-                                   "@mozilla.org/browser/history;1",
-                                   "mozIAsyncHistory");
-
-/**
- * Object that represents a mozIVisitInfo object.
- *
- * @param [optional] aTransitionType
- *        The transition type of the visit.  Defaults to TRANSITION_LINK if not
- *        provided.
- * @param [optional] aVisitTime
- *        The time of the visit.  Defaults to now if not provided.
- */
-function VisitInfo(aTransitionType, aVisitTime) {
-  this.transitionType =
-    aTransitionType === undefined ? TRANSITION_LINK : aTransitionType;
-  this.visitDate = aVisitTime || Date.now() * 1000;
-}
-
-function run_test() {
-  do_test_pending();
-
-  // Init expiration before "importing".
-  force_expiration_start();
-
-  // Add a bunch of pages (at laast IMPORT_PAGES_THRESHOLD pages).
-  let places = [];
-  for (let i = 0; i < 100; i++) {
-    places.push({
-      uri: NetUtil.newURI(TEST_URL + i),
-      title: "Title" + i,
-      visits: [new VisitInfo]
-    });
-  }
-  gHistory.updatePlaces(places);
-
-  // Set interval to a small value to expire on it.
-  setInterval(1); // 1s
-
-  Services.obs.addObserver(function observeExpiration(aSubject, aTopic, aData) {
-    Services.obs.removeObserver(observeExpiration,
-                                PlacesUtils.TOPIC_EXPIRATION_FINISHED);
-
-    // Check that statistica are up-to-date.
-    let stmt = DBConn().createAsyncStatement(
-      "SELECT (SELECT COUNT(*) FROM moz_places) - "
-      + "(SELECT SUBSTR(stat,1,LENGTH(stat)-2) FROM sqlite_stat1 "
-      + "WHERE idx = 'moz_places_url_hashindex')"
-    );
-    stmt.executeAsync({
-      handleResult(aResultSet) {
-        let row = aResultSet.getNextRow();
-        this._difference = row.getResultByIndex(0);
-      },
-      handleError(aError) {
-        do_throw("Unexpected error (" + aError.result + "): " + aError.message);
-      },
-      handleCompletion(aReason) {
-        do_check_true(this._difference === 0);
-        do_test_finished();
-      }
-    });
-    stmt.finalize();
-  }, PlacesUtils.TOPIC_EXPIRATION_FINISHED);
-}
--- a/toolkit/components/places/tests/expiration/xpcshell.ini
+++ b/toolkit/components/places/tests/expiration/xpcshell.ini
@@ -1,18 +1,16 @@
 [DEFAULT]
 head = head_expiration.js
 skip-if = toolkit == 'android'
 
-[test_analyze_runs.js]
 [test_annos_expire_history.js]
 [test_annos_expire_never.js]
 [test_annos_expire_policy.js]
 [test_annos_expire_session.js]
 [test_clearHistory.js]
 [test_debug_expiration.js]
 [test_idle_daily.js]
 [test_notifications.js]
 [test_notifications_onDeleteURI.js]
 [test_notifications_onDeleteVisits.js]
-[test_outdated_analyze.js]
 [test_pref_interval.js]
 [test_pref_maxpages.js]
deleted file mode 100644
--- a/toolkit/components/places/tests/unit/test_analyze.js
+++ /dev/null
@@ -1,28 +0,0 @@
-/* Any copyright is dedicated to the Public Domain.
-   http://creativecommons.org/publicdomain/zero/1.0/ */
-
-// Tests sqlite_sta1 table exists, it should be created by analyze.
-// Since the bookmark roots are created when the DB is created (bug 704855),
-// the table will contain data.
-
-function run_test() {
-  do_test_pending();
-
-  let stmt = DBConn().createAsyncStatement(
-    "SELECT ROWID FROM sqlite_stat1"
-  );
-  stmt.executeAsync({
-    _gotResult: false,
-    handleResult(aResultSet) {
-      this._gotResult = true;
-    },
-    handleError(aError) {
-      do_throw("Unexpected error (" + aError.result + "): " + aError.message);
-    },
-    handleCompletion(aReason) {
-      do_check_true(this._gotResult);
-       do_test_finished();
-    }
-  });
-  stmt.finalize();
-}
--- a/toolkit/components/places/tests/unit/xpcshell.ini
+++ b/toolkit/components/places/tests/unit/xpcshell.ini
@@ -51,17 +51,16 @@ skip-if = os == "linux"
 [test_463863.js]
 [test_485442_crash_bug_nsNavHistoryQuery_GetUri.js]
 [test_486978_sort_by_date_queries.js]
 [test_536081.js]
 [test_1085291.js]
 [test_1105208.js]
 [test_1105866.js]
 [test_adaptive_bug527311.js]
-[test_analyze.js]
 [test_annotations.js]
 [test_asyncExecuteLegacyQueries.js]
 [test_async_in_batchmode.js]
 [test_async_transactions.js]
 skip-if = (os == "win" && os_version == "5.1") # Bug 1158887
 [test_autocomplete_stopSearch_no_throw.js]
 [test_bookmark_catobs.js]
 [test_bookmarks_json.js]
--- a/toolkit/components/telemetry/moz.build
+++ b/toolkit/components/telemetry/moz.build
@@ -96,16 +96,20 @@ EXTRA_JS_MODULES += [
     'ThirdPartyCookieProbe.jsm',
     'UITelemetry.jsm',
 ]
 
 TESTING_JS_MODULES += [
   'tests/unit/TelemetryArchiveTesting.jsm',
 ]
 
+PYTHON_UNITTEST_MANIFESTS += [
+    'tests/python/python.ini',
+]
+
 GENERATED_FILES = [
     'TelemetryEventData.h',
     'TelemetryEventEnums.h',
     'TelemetryHistogramData.inc',
     'TelemetryHistogramEnums.h',
     'TelemetryProcessData.h',
     'TelemetryProcessEnums.h',
     'TelemetryScalarData.h',
new file mode 100644
--- /dev/null
+++ b/toolkit/components/telemetry/tests/python/python.ini
@@ -0,0 +1,1 @@
+[test_histogramtools_non_strict.py]
new file mode 100644
--- /dev/null
+++ b/toolkit/components/telemetry/tests/python/test_histogramtools_non_strict.py
@@ -0,0 +1,78 @@
+# 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/.import json
+
+import json
+import mozunit
+import sys
+import unittest
+from os import path
+
+TELEMETRY_ROOT_PATH = path.abspath(path.join(path.dirname(__file__), path.pardir, path.pardir))
+sys.path.append(TELEMETRY_ROOT_PATH)
+import histogram_tools   # noqa: E402
+
+
+def load_histogram(histograms):