Merge inbound to central, a=merge CLOSED TREE
authorWes Kocher <wkocher@mozilla.com>
Thu, 09 Mar 2017 16:33:31 -0800
changeset 496313 35398cae65c1526ce45c23a5f8b5568c5ada4762
parent 496307 fa55f8f56a1d5e64c3682b500127786082354691 (current diff)
parent 496312 f584babc1167b2fc9b1f747cb0437eab4d882514 (diff)
child 496314 e9d4b21b4d3953dc493807750d93cbe6b4746860
child 496318 54dd7a409e49edf905571a0e3661cc9565fb35fd
child 496334 2428ea9d0c062ceeff2d4ffc045deedfe4735533
child 496336 06328362721a1629dd00c3a5ce2a9dcfb0dd0296
child 496361 89d8af6c2dc2e1bfafc0edbd9e734cd32b3699bc
child 496372 b30d81e65ea580975885b4e3cfbd5614972e6f95
child 496374 d21df9bf287435d5993ac50cace666951b57b6c6
child 496375 1af80f0d1d9eee9f43bb89d2763f87279a784349
child 496387 63b32d9c8c36649b524ec22f0736c8a4f3c9885f
child 496390 eb1a2c0396e1a059285d96552062c9ab47aa2bb0
child 496415 cc38781741fc63cfefe57170db9825a1d3296d72
child 496440 6b88dffd496f48c783d547e2fc40dcd6a9034378
child 496457 ce1a7e6ac8ee7d2111f6a616d5510b70d6213563
child 496474 7d4634dc9bbe8c257d9ef2f4643b5d94b49860ee
child 496476 d0386c6252b5f89b5047f31fd2da00ab9f4bfde2
child 496480 dee3d3557c0488e8c8662717de151aa8803d697d
child 496493 3af444933a429e3d62ebfc509748dbf1da484932
child 496558 62665abe3bd5e262bd99085930c45a41e4981bf1
child 496559 9249ffed70cd5320741284a694a835c50a39a6d3
child 496574 f9b1ba831c5d06056490082f67126053c8d5002d
child 496575 5dce426312525ba4b7d31b37b28440c4c8fd0984
child 496576 b1c1e193a79688c23dd2ce6df41f6b97ce74f2d1
child 496578 dcc7c915aa8ad70f98afe28e7e7d607d01cc7eae
child 496653 52899ca960f510211e4f0108e520fc083f98ed1d
child 496659 eca5cc712358432c302fc81a47b3b8d3ece7372f
child 496679 71ea1b046d226022126272b7e4885d3ccec0f22c
child 496701 bc2ebeee788275ab2521ee9465be3d77f4c3160c
child 496702 4d233067a5a28f65c91e67bfea0a3180ba6edaad
child 496708 4c539a88fe566efadeac9ef6220d45adf2b69270
child 496984 bf8b8e497a19f7cf3ec9cd2062d3271a7448d860
child 497383 f69e0b9a8b66f6a64fe623f0a9c5b407b9432e58
child 497430 3b82cbf0c72cd9779869b000df08bce6e8e763ac
child 497467 8db212ff14aee3f1b037addd4ab2fcdb12e82d40
child 497484 d7f412e7aaf563b6083c0f5811f52605ccb4c5da
child 497670 58c2bf82edd650193b2fb4728ef2fb5780b97f5f
child 497914 583c16ac1b2f730317e119bb4d856b6ceb6f4c19
child 497950 dc8e042bd1d9450ba2f5489305df5390c1c7fe75
child 497951 75bb50c75fa1d539a1de1182e75dca25fca721db
child 497952 257611d59df0545d6b88c6b40158a231964baa21
child 498045 f728bcd907c95b7eaa4a34f9c4a0f8acc9be9043
child 498046 d77c33e0ae97d2c1cc93d804fa9171f333111791
child 498099 c5f1247dff1e73da94ebb6b26f2d7db9ae04f2a3
child 498733 b28fa4e5b2faa08ba4b5619c5109f16bc4c792ac
child 499002 59dce5c3dc4a0c740c9f6bfe5c63803195996344
child 499010 2cb303d96f185f159cd74111e502808a22a2ae42
child 499031 8e0b8906ed0c6d6ebe1b482fdd7490f2fd9b4500
child 499127 85bd36dd5f580ee94b5191e2776a9709dead4f98
child 499377 bcaad550cd35c4f9e8a5fa99ce64c8bdb26bc3a7
child 499461 e95aff1deeead5e285da382917116e7be496d100
child 499479 587572cc17301f7ba3598acc719f5f8ee9eb6bcc
child 499682 85a48c5b93b9b15f6eea96313815aaf61a4ff14b
child 499684 6d273599e666d7087f2dad5a1051bbe36df120bc
child 499748 a842e6a1383229065777a9e7181ceb8b93718a54
child 499749 f1c84306301299c3dd64763af68caa12f514750b
child 499773 00e3c1a96808099073d1cb28e4c697584630e599
child 499803 44d15ea182b23a82bc24da743aca8bf1dc04100d
child 499812 8da7950949f86e807649a1d63a9b690a1a9a77d0
child 499821 f4ec55a41030271417f8d49d981e0b05db6e79a8
child 499823 9288615c6882a91ff329a6e01632fb97b9db5092
child 500662 96dfc6bff9514e6715c2c3d21f406204dcd35036
child 501402 5aa9964c7c892e668d2c5f4382089d2399d140a3
child 501470 05cd141db1e43f603208d34890638e72ab326213
child 501885 ac505291dfdda336a208bcd88508ebfed5f8283c
child 558461 816d53ab6d31e0f7b19e3ad2f7361fec820cd101
child 565784 6f16b8b9c1ce128fdae2c3ab9dd6df27c4234ffe
push id48565
push userydelendik@mozilla.com
push dateFri, 10 Mar 2017 01:17:14 +0000
reviewersmerge
milestone55.0a1
Merge inbound to central, a=merge CLOSED TREE MozReview-Commit-ID: Gj4s6DtqKmw
devtools/client/inspector/grids/grid-inspector.js
devtools/server/actors/highlighters/css-grid.js
dom/base/nsContentUtils.cpp
dom/base/nsContentUtils.h
dom/webidl/ServiceWorkerMessageEvent.webidl
netwerk/protocol/http/nsHttpConnectionMgr.cpp
security/patches/README
testing/web-platform/meta/html/semantics/forms/textfieldselection/textfieldselection-setRangeText.html.ini
--- a/accessible/generic/Accessible.cpp
+++ b/accessible/generic/Accessible.cpp
@@ -2126,24 +2126,21 @@ Accessible::InsertChildAt(uint32_t aInde
 
   aChild->BindToParent(this, aIndex);
   return true;
 }
 
 bool
 Accessible::RemoveChild(Accessible* aChild)
 {
-  if (!aChild)
-    return false;
-
-  if (aChild->mParent != this || aChild->mIndexInParent == -1)
-    return false;
-
-  MOZ_ASSERT((mStateFlags & eKidsMutating) || aChild->IsDefunct() || aChild->IsDoc(),
-             "Illicit children change");
+  MOZ_DIAGNOSTIC_ASSERT(aChild, "No child was given");
+  MOZ_DIAGNOSTIC_ASSERT(aChild->mParent == this, "Wrong parent");
+  MOZ_DIAGNOSTIC_ASSERT(aChild->mIndexInParent != -1, "Unbound child was given");
+  MOZ_DIAGNOSTIC_ASSERT((mStateFlags & eKidsMutating) || aChild->IsDefunct() || aChild->IsDoc(),
+                        "Defunct child was given");
 
   int32_t index = static_cast<uint32_t>(aChild->mIndexInParent);
   if (mChildren.SafeElementAt(index) != aChild) {
     MOZ_ASSERT_UNREACHABLE("A wrong child index");
     index = mChildren.IndexOf(aChild);
     if (index == -1) {
       MOZ_ASSERT_UNREACHABLE("No child was found");
       return false;
--- a/accessible/xul/XULTreeAccessible.cpp
+++ b/accessible/xul/XULTreeAccessible.cpp
@@ -860,16 +860,17 @@ XULTreeItemAccessibleBase::DoAction(uint
 // XULTreeItemAccessibleBase: Accessible implementation
 
 void
 XULTreeItemAccessibleBase::Shutdown()
 {
   mTree = nullptr;
   mTreeView = nullptr;
   mRow = -1;
+  mParent = nullptr; // null-out to prevent base class's shutdown ops
 
   AccessibleWrap::Shutdown();
 }
 
 GroupPos
 XULTreeItemAccessibleBase::GroupPosition()
 {
   GroupPos groupPos;
--- a/accessible/xul/XULTreeGridAccessible.cpp
+++ b/accessible/xul/XULTreeGridAccessible.cpp
@@ -450,16 +450,28 @@ NS_IMPL_CYCLE_COLLECTION_INHERITED(XULTr
 NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION_INHERITED(XULTreeGridCellAccessible)
 NS_INTERFACE_MAP_END_INHERITING(LeafAccessible)
 NS_IMPL_ADDREF_INHERITED(XULTreeGridCellAccessible, LeafAccessible)
 NS_IMPL_RELEASE_INHERITED(XULTreeGridCellAccessible, LeafAccessible)
 
 ////////////////////////////////////////////////////////////////////////////////
 // XULTreeGridCellAccessible: Accessible
 
+void
+XULTreeGridCellAccessible::Shutdown()
+{
+  mTree = nullptr;
+  mTreeView = nullptr;
+  mRow = -1;
+  mColumn = nullptr;
+  mParent = nullptr; // null-out to prevent base class's shutdown ops
+
+  LeafAccessible::Shutdown();
+}
+
 Accessible*
 XULTreeGridCellAccessible::FocusedChild()
 {
   return nullptr;
 }
 
 ENameValueFlag
 XULTreeGridCellAccessible::Name(nsString& aName)
--- a/accessible/xul/XULTreeGridAccessible.h
+++ b/accessible/xul/XULTreeGridAccessible.h
@@ -119,16 +119,17 @@ public:
                             int32_t aRow, nsITreeColumn* aColumn);
 
   // nsISupports
   NS_DECL_ISUPPORTS_INHERITED
   NS_DECL_CYCLE_COLLECTION_CLASS_INHERITED(XULTreeGridCellAccessible,
                                            LeafAccessible)
 
   // Accessible
+  virtual void Shutdown() override;
   virtual TableCellAccessible* AsTableCell() override { return this; }
   virtual nsIntRect Bounds() const override;
   virtual ENameValueFlag Name(nsString& aName) override;
   virtual Accessible* FocusedChild() override;
   virtual already_AddRefed<nsIPersistentProperties> NativeAttributes() override;
   virtual int32_t IndexInParent() const override;
   virtual Relation RelationByType(RelationType aType) override;
   virtual a11y::role NativeRole() override;
--- a/browser/components/search/test/browser_amazon.js
+++ b/browser/components/search/test/browser_amazon.js
@@ -28,17 +28,17 @@ function test() {
   const EXPECTED_ENGINE = {
     name: "Amazon.com",
     alias: null,
     description: "Amazon.com Search",
     searchForm: "https://www.amazon.com/exec/obidos/external-search/?field-keywords=&ie=UTF-8&mode=blended&tag=mozilla-20&sourceid=Mozilla-search",
     hidden: false,
     wrappedJSObject: {
       queryCharset: "UTF-8",
-      "_iconURL": "",
+      "_iconURL": "resource://search-plugins/images/amazon.ico",
       _urls : [
         {
           type: "application/x-suggestions+json",
           method: "GET",
           template: "https://completion.amazon.com/search/complete?q={searchTerms}&search-alias=aps&mkt=1",
           params: "",
         },
         {
--- a/browser/components/search/test/browser_yahoo.js
+++ b/browser/components/search/test/browser_yahoo.js
@@ -42,17 +42,17 @@ function test() {
   const EXPECTED_ENGINE = {
     name: "Yahoo",
     alias: null,
     description: "Yahoo Search",
     searchForm: "https://search.yahoo.com/yhs/search?p=&ei=UTF-8&hspart=mozilla&hsimp=yhs-001",
     hidden: false,
     wrappedJSObject: {
       queryCharset: "UTF-8",
-      "_iconURL": "",
+      "_iconURL": "resource://search-plugins/images/yahoo.ico",
       _urls : [
         {
           type: "application/x-suggestions+json",
           method: "GET",
           template: "https://search.yahoo.com/sugg/ff",
           params: [
             {
               name: "output",
--- a/browser/locales/jar.mn
+++ b/browser/locales/jar.mn
@@ -88,17 +88,19 @@
 % resource search-plugins chrome://browser/locale/searchplugins/
 #if BUILD_FASTER
     locale/browser/searchplugins/               (searchplugins/*.xml)
     locale/browser/searchplugins/list.json      (search/list.json)
 #else
     locale/browser/searchplugins/               (.deps/generated_@AB_CD@/*.xml)
     locale/browser/searchplugins/list.json      (.deps/generated_@AB_CD@/list.json)
 #endif
+    locale/browser/searchplugins/images/amazon.ico     (searchplugins/images/amazon.ico)
     locale/browser/searchplugins/images/wikipedia.ico  (searchplugins/images/wikipedia.ico)
+    locale/browser/searchplugins/images/yahoo.ico      (searchplugins/images/yahoo.ico)
     locale/browser/searchplugins/images/yandex-en.ico  (searchplugins/images/yandex-en.ico)
 % locale browser-region @AB_CD@ %locale/browser-region/
     locale/browser-region/region.properties        (%chrome/browser-region/region.properties)
 # the following files are browser-specific overrides
     locale/browser/netError.dtd                (%chrome/overrides/netError.dtd)
     locale/browser/appstrings.properties       (%chrome/overrides/appstrings.properties)
     locale/browser/downloads/settingsChange.dtd  (%chrome/overrides/settingsChange.dtd)
 % override chrome://global/locale/netError.dtd chrome://browser/locale/netError.dtd
--- a/browser/locales/searchplugins/amazon-en-GB.xml
+++ b/browser/locales/searchplugins/amazon-en-GB.xml
@@ -1,17 +1,17 @@
 <!-- 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>Amazon.co.uk</ShortName>
 <Description>Amazon.co.uk Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/amazon.ico</Image>
 <Url type="text/html" method="GET" template="https://www.amazon.co.uk/exec/obidos/external-search/" resultdomain="amazon.co.uk">
   <Param name="field-keywords" value="{searchTerms}"/>
   <Param name="ie" value="{inputEncoding}"/>
   <Param name="mode" value="blended"/>
   <Param name="tag" value="firefox-uk-21"/>
   <Param name="sourceid" value="Mozilla-search"/>
 </Url>
 <SearchForm>https://www.amazon.co.uk/</SearchForm>
--- a/browser/locales/searchplugins/amazon-france.xml
+++ b/browser/locales/searchplugins/amazon-france.xml
@@ -1,17 +1,17 @@
 <!-- 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>Amazon.fr</ShortName>
 <Description>Recherche Amazon.fr</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/amazon.ico</Image>
 <Url type="text/html" method="GET" template="https://www.amazon.fr/exec/obidos/external-search/" resultdomain="amazon.fr">
   <Param name="field-keywords" value="{searchTerms}"/>
   <Param name="ie" value="{inputEncoding}"/>
   <Param name="mode" value="blended"/>
   <Param name="tag" value="firefox-fr-21"/>
   <Param name="sourceid" value="Mozilla-search"/>
 </Url>
 <SearchForm>https://www.amazon.fr/</SearchForm>
--- a/browser/locales/searchplugins/amazon-in.xml
+++ b/browser/locales/searchplugins/amazon-in.xml
@@ -1,17 +1,17 @@
 <!-- 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>Amazon.in</ShortName>
 <Description>Amazon.in Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/amazon.ico</Image>
 <Url type="text/html" method="GET" template="https://www.amazon.in/exec/obidos/external-search/" resultdomain="amazon.in">
   <Param name="field-keywords" value="{searchTerms}"/>
   <Param name="ie" value="{inputEncoding}"/>
   <Param name="mode" value="blended"/>
   <Param name="sourceid" value="Mozilla-search"/>
 </Url>
 <SearchForm>https://www.amazon.in/</SearchForm>
 </SearchPlugin>
--- a/browser/locales/searchplugins/amazon-it.xml
+++ b/browser/locales/searchplugins/amazon-it.xml
@@ -1,17 +1,17 @@
 <!-- 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>Amazon.it</ShortName>
 <Description>Ricerca Amazon.it</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/amazon.ico</Image>
 <Url type="text/html" method="GET" template="https://www.amazon.it/exec/obidos/external-search/" resultdomain="amazon.it">
   <Param name="field-keywords" value="{searchTerms}"/>
   <Param name="ie" value="{inputEncoding}"/>
   <Param name="mode" value="blended"/>
   <Param name="tag" value="firefoxit-21"/>
   <Param name="sourceid" value="Mozilla-search"/>
 </Url>
 <SearchForm>https://www.amazon.it/</SearchForm>
--- a/browser/locales/searchplugins/amazon-jp.xml
+++ b/browser/locales/searchplugins/amazon-jp.xml
@@ -1,17 +1,17 @@
 <!-- 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>Amazon.co.jp</ShortName>
 <Description>Amazon.co.jp Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/amazon.ico</Image>
 <Url type="text/html" method="GET" template="https://www.amazon.co.jp/exec/obidos/external-search/" resultdomain="amazon.co.jp">
   <Param name="field-keywords" value="{searchTerms}"/>
   <Param name="mode" value="blended"/>
   <!--
   <Param name="mode" value="books-jp"/>
   <Param name="mode" value="books-us"/>
   -->
   <Param name="tag" value="mozillajapan-fx-22"/>
--- a/browser/locales/searchplugins/amazondotcn.xml
+++ b/browser/locales/searchplugins/amazondotcn.xml
@@ -1,17 +1,17 @@
 <!-- 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>亚马逊</ShortName>
 <Description>亚马逊搜索</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/amazon.ico</Image>
 <Url type="text/html" method="GET" template="https://www.amazon.cn/mn/searchApp" resultdomain="amazon.cn">
   <Param name="keywords" value="{searchTerms}"/>
   <Param name="ix" value="sunray"/>
   <Param name="pageletid" value="headsearch"/>
   <Param name="searchType" value=""/>
   <Param name="Go.x" value="0"/>
   <Param name="Go.y" value="0"/>
   <Param name="bestSaleNum" value="0"/>
--- a/browser/locales/searchplugins/amazondotcom-de.xml
+++ b/browser/locales/searchplugins/amazondotcom-de.xml
@@ -1,17 +1,17 @@
 <!-- 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>Amazon.de</ShortName>
 <Description>Amazon.de Suche</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/amazon.ico</Image>
 <Url type="text/html" method="GET" template="https://www.amazon.de/exec/obidos/external-search/" resultdomain="amazon.de">
   <Param name="field-keywords" value="{searchTerms}"/>
   <Param name="ie" value="{inputEncoding}"/>
   <Param name="mode" value="blended"/>
   <Param name="tag" value="firefox-de-21"/>
   <Param name="sourceid" value="Mozilla-search"/>
 </Url>
 <SearchForm>https://www.amazon.de/</SearchForm>
--- a/browser/locales/searchplugins/amazondotcom.xml
+++ b/browser/locales/searchplugins/amazondotcom.xml
@@ -1,17 +1,17 @@
 <!-- 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>Amazon.com</ShortName>
 <Description>Amazon.com Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/amazon.ico</Image>
 <Url type="application/x-suggestions+json" method="GET" template="https://completion.amazon.com/search/complete?q={searchTerms}&amp;search-alias=aps&amp;mkt=1"/>
 <Url type="text/html" method="GET" template="https://www.amazon.com/exec/obidos/external-search/" rel="searchform">
   <Param name="field-keywords" value="{searchTerms}"/>
   <Param name="ie" value="{inputEncoding}"/>
   <Param name="mode" value="blended"/>
   <Param name="tag" value="mozilla-20"/>
   <Param name="sourceid" value="Mozilla-search"/>
 </Url>
new file mode 100644
index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..1c39eaf8fee5b2a29abbc32f85f4187e02f7114a
GIT binary patch
literal 1407
zc$@)$1%Ubh0096301yxW009690IdN4044wc03aX$0096X0IUN50Mh{g0EtjeM-2)Z
z3IG5A4M|8uQUCw|5C8xG5C{eU001BJ|6u?C0dz@3K~y-6rBgji!%!4Gk-RR72u_ve
z2UHvyCl{xXtqv;b*42(ehyDY1p;?QQ#YHQ)3c4vJQ>kFOScj%qmxi=UD;$T^(0ph`
z58UD1ckg-U-1nX&3xVlmLi_<Dm69ZqvF1*vgIcYImu?pTu$RxHSS%*9Fapc6C<_sb
zN~IDlB}v8zj!GrULQvCn+O{36Gm4T(5VO$l_u)7WT-Sx?c{s1v(P%UP0Mp52u?5?<
znN!t78^5w(6bh7uU`|!(I1a0(`6IznRMD4Q!!VFer_pM)g1ymbwAjMDd`;Iw7OJNC
zjfHV9NLZG|jG_dSubL)}LV=pD)AKyCb4E7Hegp(JS#tQC=rM%5eZTSPl)S(kLGRX2
zG|M41F^qd;_sT!)a-HP-J9upXj|cu>)^-8qFg&ccdth)4>|6n26&SXG&0~Zg(z^w;
zPyE6tmb1Wi71+>#kJ)X=D@goIaBX1l08HJPqAkBM0n&fJ^C!nu=QNrtGARH6002ov
zPDHLkV1kKIPDc$28VUda01Zh<L{b0%03ZMW03Zkk0001UdV2H#00VPLL_t(og|(MY
zXcSi*$3L&`29#hM%2pO#kx*zOt?VR<|L94#_0pmR1HJUn(Cop3Xkczp5EKt$E#|HS
zJeW`&@zP8BCrWGL21~kaG8Bq!-Cf*JnlhUl65;1zGI1yScIG7^Ul`c;o8P?OXMXSZ
zdv6w&BqEzdq)$Ztkd4#iQjE>iOGMf?IIs?+9Y8J7OagCXs9LRt>$=Nhr4;FOnshpC
zIt9S5z<xyJoGG^BIC3<XlSDG9y^iK`;yBJ~GSP<^q~7zq$iOQv+qO*s3Yd7E?CuT)
z*px`bLKtRZJ{G`n9BqKCWy#6z?xvYJlFOL@NG!mYy}hw1wryJrVWQXKN>wS<VC(Pi
z4~=_zdR9~Y;rT^G;+*Q+PANraXJ>;?vZbZLwz#km3y?6OY;A3&tE+2yT&Ywjm&^S8
z=n(*;qoV+bh&E&E1_*-QxOtP|;bDr!qSj}%!2yig&~Y5uk;xeE1FI>**czCanBc&{
zgW7v%&zvE%b0@nyI{4_=v4;C5H4vMDtYvAp+)}9&@>^>LA^}|2)lOU}6heOGa(QF+
zfvKq}t-Zb@qWmhAO6&?m0{rvrnbuyb)k5RjLql4>@$vCkfXF2n85z+YKJ{Di;NW1>
z11f7-;(1==5<~(tTialAlh(XGFtBz2k;bDq3TUNmM<ye#>#h%=*~dtI%5B>g&+|g3
zV%D-^gBt+^5Z4C=mO(Zp5|Xtn>CNXGTJqwhOVXRqhu40tBvO4WQXLmr4A(?>C;a_u
z*ph)=pT<|OiRQ~Nb_%Kv1X1=}gWac_biwSj$Tx3_e0SU@UauBCf1bIL|NDN2$Zyv|
zF5!LP^>^Ttr?BuZjO>H&Pr%bV@FLvz#;T4%@gm&c4s#`_O+o5V$VGn`y{N&H^YG`@
zSE<W<0d0FAbr{;-g+y!PGyhG)Y!Rvt;Q2lOMHF!04kWii@dAAKWBpVAD-oFl-q*gt
zOaaC|_sN5!ekHD|v#tCb*mfMO%YJ5O9z)Ak|Mx3BR3g#`T+spq`s*%Ce+9D-{9r*)
z9h3sv&%&<HAhkDgS|A_%HTwz3#(6VehM6&#`x~DBuoO^1vdv$fH{OG{_BSq%UJsR0
zANVgIk`j@7>o0p_-4c;hkg#Qm$O+(MU@y(;V(t0`cmfP5rM?Y}F99d;KGOI<2~Ge2
N002ovPDHLkV1jb8k9+_C
new file mode 100644
index e69de29bb2d1d6434b8b29ae775ad8c2e48c5391..9bd1d9f7c008c3f288c759f70b42a6f32bf55211
GIT binary patch
literal 5430
zc$~FZX;4<z89sOWzK@U4m|vYvlc|5&X*>DRi3qrW;(`dWsDJ|6rcF9clL>CPfTFkt
z6jwl8BH)IC;2wj57*|{xHBpmk7SkVDnl8qPOWdD6@44T1A#pnS(e%zdbMHOpec$Ij
z=brPP_q|FP%2YjjD&Bjl?LCzGg;J_V5BL2gTdDJW{w3Qeq6R9rU9Qp|>alBq>W7I|
z@6!|PnAavaF+*Ndzi}Ry`e81yEgYK5itZF{&98c40#QIrQoZgL7`;B3Y{kA+VE0}_
z`Kc5AahU~9?63Z;o{RcEj#pK$$O0$k*ORC-!HF%Tjr9dqbO&u-)wV-C!WrggJEo{;
zjcKYk-=i^^kozP<z&10C==&3`=-ZR6-q(1)HpxTpo6<Mg>?SLEqEGu`Zz^q1(YC+M
zU&!|9gm4$B7%ey5h`|ga7BkFPOgH5ohayYR2gSCFX*T<KFY?~wFwKf3c&E-ZGsXpN
z6^kv}77pQ?+>KxOq;IB*)%-=2EjHpXi<~89AI!1(q15h=xx{Q@4pHj#$4sX$W;y*(
zLKO3Prp<Sv*mj?1($|^PFE#lr_K4}UUl{Ts%r8C_lY5TpgA&S=806G?<R$k!^3Av7
zG0%y|d?x`5h=tAol=|W^*O!2KM7Z7UOVIvmP=BG>9}5WKpG_2nd<gT)+za<SK9?Hg
zGhHmO`eUJ;fJFlGl{*7b?n^|OKM_|N+HmjI9gX)I53`MZY?pqK6(2-bP-Z3IHIu&6
z=6s_+N>rR4bC_SoK3Da{BK9q!pJiqO%B=xdY!AQ^ClM9IQr{pf^AFZo?jM57b#Eek
zU8;W*O9Ml6-%@`_&PO<dG{lbhu$XpbE@X~{Gt4hzUr6o>gJTSFFR=%r!cNrs%V}!`
zIam6JqB4+#D&q3G{T~074F^yi7)B)NdzGIulwIW;f>jRn>_J2#mRN~;Y{Dh$tBXGl
zWkx(!nu%C0pseVKZWV1*YkNuBeoY`5Ya)_yaoG-!|KiGK)C7~Un(eFE@A?q(!#7mh
zSw(xxWIVJlbI_fCJ|SaYZ4O42MNWG#s;N`sBw-Ehtn(*hePB3|g7v`^ydF%&I~7eH
z|2q{sP#YYf`_>U_DZ7?_tfijliyg67Z4bc;Ymnxbz3qK}`m)kUM2$&KYbe&)!?4~N
zhSz<`*x(<ITJqKfQc+Kijlnc*j7-P*(#>7tU$_nRk!h$S#E)9~w;>>)uhg&irJ&YH
z)^=rl@(t!0{XG6fe6BJFVVyY?8z@t6Cu5_Ng3bO^jV*yRY>h}qLofpkkr~()nThk$
zH+cN#XV#<PiA?roy8XdSY>J?t)T^idMxsvhCt)qIl3cPE!u&G#HJtl;D_L`IcE~C4
zr=cMrNXK^CY$Tc@M`6bkqj7P1oyUKH&rMOI(ZqJ?Zwijo*h;;v)E7H4o_a#~m-8FU
zW_$_rmm3LK!?AC%Qn1xd#Wp7m+kI(h^k<+ckcnnO>@`P@L`&po?25|5rJ_w9|GUMT
zurn%~$YOt%mJ#_z>NN&9AO3VS`O@{6WDY904`iQ*`InG?gUNZfhNHz!$4(~$yL_40
z?I!}GuqR?P_5`!AH&T#;eNkiZ)n7jJ_`iDlBkcRpSnOxJ^zV(##$L*6y#W2C9pP_s
z(oxI#t>zkA!Zpx+{x_P*Xs}W>|1O8TzLDe~sr3b7Zy)XMqy6IvMfmP@DzOBuQMtN*
zzxWu;#{NJS-XO$IGsh$R8@jE*F#j@RAhwt(XtC09z#fHGCyU6&K|c}5!GS<F+Ja+U
zMCRhwM_)X~fBWOVqwR@2v__7>!QdDiqU@o-7_?G<zkf7#J0r1!{G06*tg-kF%z+;N
za`JC8Q?b*`K%1R~b~^{{&KPv~1!K_>$i<NeBAACuITaq~{p+_h!q?^86=<h_?ZI3e
zro7gpek;eZ&*68lGqF+lO_~1z9>47UMk5t_%}gA&vhkKZ21lK-IO@yAQ9lvLa}mtP
z-7l|ueD6J9jrYb^dhhrC`8AFO$GM>Vu|PhK`SWnZ$2sAg?z6cjEyf3a>(XxfzfvWl
z#Yn?`b0m&g<kkEF;m^}PoD7V^)xxedfBVBPa3XTN#_bP2_u8+{*oqV675Ps8IGpt7
zYdakd*JX%b)_SSY*W<5NgRt93$3b&6PMTxUY3C97IPK)4)92bb9sDgG-n!cr^WtXW
zB*%F&I35>cih)Oubo;~GcX1}_d7So-C&u9n^-nwFaEdnD>}>4e97~+*9{+@Ym+%|H
zpNmc_FXRKEeaOd`wMRVe>m8T%_%tHuAMe-OFM0j{*mx9Y$bHr~9%spY+NK<B9kQ}C
z{}v;Kc|aHc>naKRR3<u%Y@9apa8_XD<1F<=|NPUh;L(GJ+P2)!{p2OhspnW@JkI_2
zPk40yf$PA-N4W6JE7TpYagKIQSX`GTW0RSVI%Al}EvQk0u~%iFU1g!u$kRTY>NXEI
z&UD3^&)2l+H6`ofghsxH&nmlek{jnf=D71+(2mS&hndaq%y_EOu!?K$Ti;)C{uSrn
z7>g6cajt`7#HEziJev=0-a$L}<RPx5gNzTYKE?-M4h{*}*2Z=I;OlQZ`FGRH^m^_j
zxA3<aS^Ul!XjG|Kp}OsVH2@7N1-s}&8|R>v>)?Pf3J16!B-Y&f_YFL}_W&1PuE8Fz
z;oSnp6CEcB_wKe|c%c#x@7%*3?$ev^e~MPF=?*hTbMGeiHgd~&%T#<<{0s35c5rTF
zE$rnyH83VLFfMH+w#c3FS<a{}Jfj2>%Qo{aF-1^s57#mh$3;)D%^ZPd?(IfngtlL!
zhG_mds%!tv56wXp+ndw~`sm7O7%o;bH>u<qy@F@=Qsz*SBg#1<v3#X9Sf8mCrklgK
za*3`Uv7KYy9vb%=+ALEEtTnpwui1Q_ADRdGE(=3BsnjF#Ecv3;Q4?8DO*DEvm|#YK
zLtI;6#$RKNaIM7bdyVzc%_(Ng1Cf;+TlTl0l;f1OCUzyx%?st*-Sc0`#YIjydA~39
zCu_!|v+REBXV`tFPxbee`t#YB)pJjNU;SC2qN57z*k`5woXFahweI>-tN#)D@(H=`
z3y*M5?Uw&fwqm6IUd&oIX`+hu|1j0|<CoZ;;>0Bt+Odm;Q@H)W-_(=W|6i2<{UWDu
z%YO>_EzwhT{ihgcKQ7L?9<RE!`*Gq$*4EDyn!S4znlVoi;rmn4#=fU1^8(Qs)|3AW
zQu9p>p^!B(Wr>*Y!0q$ohzah030*gqaE(arsKJ_d*-w+*V8ioAo@cY}_4+9*I^q8d
zrF>t+ytmAXM>*@l#jNW#Gf&p)jM<WBm$R*0*NpM38T)CSDWQJ{lBY|3RLS~8YO6}0
zTNi~dd6u?H>7$I)I-)bx{l9QyDRZvX_E6RqNxHU>+MqTvHN^i=KTEwKeIg^Wq9eL9
zRLp0U-1~KspZlaPU|t@PftH`=dj2>4EJwFVpU6nAxr#cXTcl$Dy53Ae19RYJAN80M
zwnU7=yT4!Bb*AT+V<+31b-tK^t&|lV(G}D&Z)jm|yx%tpZ}_wDW-tfey7&#GP5MNp
znfadRh%V#JXYwrhWgE{6IY-`~=hiSEEZ&Sue_WyUrR^~L+9~q}<GJXFF2Cj1a&{hO
z4sp`qxyPI!d`XTYXR-IrK9LdGX6lIUTz<zU#`ELm7=0#xa^Z@Wz4A&GPBB*ygy-Vz
z@=f~e7ujvh73S)H6Mc9#r0U%8#P4Q#T*o9w^o`Y!b2)sS_(KVf^4t*}(UsU%&pcxf
z&y{N*eX3=e?F{T-zTe1tM{;v{ZegAdrc(d9d=-0`6V$4qT37xLE#a9ZXGAsEnB?qb
zX1s>fvkUcq3-?|6Dk&$ji>V{J?jB?eSJB6(sNOF!|9pnBJp|MhJjG{059WqXQ|1NA
Mc8ZSZ{;%QYKfucDssI20
--- a/browser/locales/searchplugins/yahoo-NO.xml
+++ b/browser/locales/searchplugins/yahoo-NO.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo Søk</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://no.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://no.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-ar.xml
+++ b/browser/locales/searchplugins/yahoo-ar.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo Argentina</ShortName>
 <Description>Buscar en Yahoo Argentina</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://ar.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://ar.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-br.xml
+++ b/browser/locales/searchplugins/yahoo-br.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Pesquisa Yahoo</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://br.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://br.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-ch.xml
+++ b/browser/locales/searchplugins/yahoo-ch.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Tschertga Yahoo Svizra</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://ch.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://ch.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-cl.xml
+++ b/browser/locales/searchplugins/yahoo-cl.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Buscar en Yahoo Chile</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://cl.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://cl.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-de.xml
+++ b/browser/locales/searchplugins/yahoo-de.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://de.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://de.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-en-CA.xml
+++ b/browser/locales/searchplugins/yahoo-en-CA.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo Canada</ShortName>
 <Description>Yahoo Canada Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://ca.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://ca.search.yahoo.com/yhs/search"
      resultdomain="yahoo.com" rel="searchform">
--- a/browser/locales/searchplugins/yahoo-en-GB.xml
+++ b/browser/locales/searchplugins/yahoo-en-GB.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo.co.uk</ShortName>
 <Description>Yahoo UK &amp; Ireland Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://uk.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://uk.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-es.xml
+++ b/browser/locales/searchplugins/yahoo-es.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://es.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://es.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-espanol.xml
+++ b/browser/locales/searchplugins/yahoo-espanol.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://espanol.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://espanol.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-fi.xml
+++ b/browser/locales/searchplugins/yahoo-fi.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo-haku</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://fi.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://fi.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-france.xml
+++ b/browser/locales/searchplugins/yahoo-france.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Recherche Yahoo</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://fr.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://fr.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-fy-NL.xml
+++ b/browser/locales/searchplugins/yahoo-fy-NL.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Sykje mei Yahoo</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://nl.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://nl.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-id.xml
+++ b/browser/locales/searchplugins/yahoo-id.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Pencarian Yahoo</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://id.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://id.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-in.xml
+++ b/browser/locales/searchplugins/yahoo-in.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://in.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://in.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-it.xml
+++ b/browser/locales/searchplugins/yahoo-it.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://it.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://it.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-mx.xml
+++ b/browser/locales/searchplugins/yahoo-mx.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://mx.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://mx.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-sv-SE.xml
+++ b/browser/locales/searchplugins/yahoo-sv-SE.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo Sök</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://se.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://se.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-tl.xml
+++ b/browser/locales/searchplugins/yahoo-tl.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://sg.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://sg.search.yahoo.com/search" resultdomain="yahoo.com">
   <Param name="p" value="{searchTerms}"/>
--- a/browser/locales/searchplugins/yahoo-zh-TW-HK.xml
+++ b/browser/locales/searchplugins/yahoo-zh-TW-HK.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo 雅虎香港</ShortName>
 <Description>Yahoo 雅虎香港</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://hk.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://hk.search.yahoo.com/yhs/search"
      resultdomain="yahoo.com" rel="searchform">
--- a/browser/locales/searchplugins/yahoo-zh-TW.xml
+++ b/browser/locales/searchplugins/yahoo-zh-TW.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo 搜尋</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://tw.search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://tw.search.yahoo.com/yhs/search"
      resultdomain="yahoo.com" rel="searchform">
--- a/browser/locales/searchplugins/yahoo.xml
+++ b/browser/locales/searchplugins/yahoo.xml
@@ -1,17 +1,17 @@
 <!-- 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>Yahoo</ShortName>
 <Description>Yahoo Search</Description>
 <InputEncoding>UTF-8</InputEncoding>
-<Image width="16" height="16"></Image>
+<Image width="16" height="16">resource://search-plugins/images/yahoo.ico</Image>
 <Url type="application/x-suggestions+json" method="GET"
      template="https://search.yahoo.com/sugg/ff">
   <Param name="output"  value="fxjson" />
   <Param name="appid"   value="ffd" />
   <Param name="command" value="{searchTerms}" />
 </Url>
 <Url type="text/html" method="GET" template="https://search.yahoo.com/yhs/search"
      resultdomain="yahoo.com" rel="searchform">
--- a/devtools/client/inspector/grids/components/Grid.js
+++ b/devtools/client/inspector/grids/components/Grid.js
@@ -23,16 +23,17 @@ module.exports = createClass({
     grids: PropTypes.arrayOf(PropTypes.shape(Types.grid)).isRequired,
     highlighterSettings: PropTypes.shape(Types.highlighterSettings).isRequired,
     setSelectedNode: PropTypes.func.isRequired,
     showGridOutline: PropTypes.bool.isRequired,
     onHideBoxModelHighlighter: PropTypes.func.isRequired,
     onSetGridOverlayColor: PropTypes.func.isRequired,
     onShowBoxModelHighlighterForNode: PropTypes.func.isRequired,
     onShowGridAreaHighlight: PropTypes.func.isRequired,
+    onShowGridCellHighlight: PropTypes.func.isRequired,
     onToggleGridHighlighter: PropTypes.func.isRequired,
     onToggleShowGridLineNumbers: PropTypes.func.isRequired,
     onToggleShowInfiniteLines: PropTypes.func.isRequired,
   },
 
   mixins: [ addons.PureRenderMixin ],
 
   render() {
@@ -44,27 +45,29 @@ module.exports = createClass({
       showGridOutline,
       onHideBoxModelHighlighter,
       onSetGridOverlayColor,
       onShowBoxModelHighlighterForNode,
       onToggleGridHighlighter,
       onToggleShowGridLineNumbers,
       onToggleShowInfiniteLines,
       onShowGridAreaHighlight,
+      onShowGridCellHighlight,
     } = this.props;
 
     return grids.length ?
       dom.div(
         {
           id: "layout-grid-container",
         },
         showGridOutline ?
           GridOutline({
             grids,
             onShowGridAreaHighlight,
+            onShowGridCellHighlight,
           })
           :
           null,
         dom.div(
           {
             className: "grid-content",
           },
           GridList({
--- a/devtools/client/inspector/grids/components/GridOutline.js
+++ b/devtools/client/inspector/grids/components/GridOutline.js
@@ -19,16 +19,17 @@ const VIEWPORT_WIDTH = 450;
 
 module.exports = createClass({
 
   displayName: "GridOutline",
 
   propTypes: {
     grids: PropTypes.arrayOf(PropTypes.shape(Types.grid)).isRequired,
     onShowGridAreaHighlight: PropTypes.func.isRequired,
+    onShowGridCellHighlight: PropTypes.func.isRequired,
   },
 
   mixins: [ addons.PureRenderMixin ],
 
   getInitialState() {
     return {
       selectedGrids: [],
     };
@@ -70,17 +71,18 @@ module.exports = createClass({
    *
    * @param  {Object} grid
    *         A single grid container in the document.
    */
   renderGrid(grid) {
     const { id, color, gridFragments } = grid;
     // TODO: We are drawing the first fragment since only one is currently being stored.
     // In the future we will need to iterate over all fragments of a grid.
-    const { rows, cols, areas } = gridFragments[0];
+    let gridFragmentIndex = 0;
+    const { rows, cols, areas } = gridFragments[gridFragmentIndex];
     const numberOfColumns = cols.lines.length - 1;
     const numberOfRows = rows.lines.length - 1;
     const rectangles = [];
 
     // Draw a rectangle that acts as a border for the grid outline.
     const border = this.renderGridOutlineBorder(numberOfRows, numberOfColumns, color);
     rectangles.push(border);
 
@@ -88,18 +90,18 @@ module.exports = createClass({
     let y = 1;
     const width = 10;
     const height = 10;
 
     // Draw the cells within the grid outline border.
     for (let rowNumber = 1; rowNumber <= numberOfRows; rowNumber++) {
       for (let columnNumber = 1; columnNumber <= numberOfColumns; columnNumber++) {
         const gridAreaName = this.getGridAreaName(columnNumber, rowNumber, areas);
-        const gridCell = this.renderGridCell(id, x, y, rowNumber, columnNumber, color,
-          gridAreaName);
+        const gridCell = this.renderGridCell(id, gridFragmentIndex, x, y, rowNumber,
+          columnNumber, color, gridAreaName);
 
         rectangles.push(gridCell);
         x += width;
       }
 
       x = 1;
       y += height;
     }
@@ -107,33 +109,39 @@ module.exports = createClass({
     return rectangles;
   },
 
   /**
    * Renders the grid cell of a grid fragment.
    *
    * @param  {Number} id
    *         The grid id stored on the grid fragment
+   * @param  {Number} gridFragmentIndex
+   *         The index of the grid fragment rendered to the document.
    * @param  {Number} x
    *         The x-position of the grid cell.
    * @param  {Number} y
    *         The y-position of the grid cell.
    * @param  {Number} rowNumber
    *         The row number of the grid cell.
    * @param  {Number} columnNumber
    *         The column number of the grid cell.
    * @param  {String|null} gridAreaName
    *         The grid area name or null if the grid cell is not part of a grid area.
    */
-  renderGridCell(id, x, y, rowNumber, columnNumber, color, gridAreaName) {
+  renderGridCell(id, gridFragmentIndex, x, y, rowNumber, columnNumber, color,
+    gridAreaName) {
     return dom.rect(
       {
         className: "grid-outline-cell",
         "data-grid-area-name": gridAreaName,
+        "data-grid-fragment-index": gridFragmentIndex,
         "data-grid-id": id,
+        "data-grid-row": rowNumber,
+        "data-grid-column": columnNumber,
         x,
         y,
         width: 10,
         height: 10,
         fill: "none",
         stroke: color,
         onMouseOver: this.onMouseOverCell,
         onMouseOut: this.onMouseLeaveCell,
@@ -162,37 +170,50 @@ module.exports = createClass({
       }
     );
   },
 
   onMouseLeaveCell({ target }) {
     const {
       grids,
       onShowGridAreaHighlight,
+      onShowGridCellHighlight,
     } = this.props;
     const id = target.getAttribute("data-grid-id");
     const color = target.getAttribute("stroke");
 
     target.setAttribute("fill", "none");
 
     onShowGridAreaHighlight(grids[id].nodeFront, null, color);
+    onShowGridCellHighlight(grids[id].nodeFront);
   },
 
   onMouseOverCell({ target }) {
     const {
       grids,
       onShowGridAreaHighlight,
+      onShowGridCellHighlight,
     } = this.props;
     const name = target.getAttribute("data-grid-area-name");
     const id = target.getAttribute("data-grid-id");
+    const fragmentIndex = target.getAttribute("data-grid-fragment-index");
     const color = target.getAttribute("stroke");
+    const rowNumber = target.getAttribute("data-grid-row");
+    const columnNumber = target.getAttribute("data-grid-column");
 
     target.setAttribute("fill", color);
 
-    onShowGridAreaHighlight(grids[id].nodeFront, name, color);
+    if (name) {
+      onShowGridAreaHighlight(grids[id].nodeFront, name, color);
+    }
+
+    if (fragmentIndex && rowNumber && columnNumber) {
+      onShowGridCellHighlight(grids[id].nodeFront, fragmentIndex,
+        rowNumber, columnNumber);
+    }
   },
 
   render() {
     return this.state.selectedGrids.length ?
       dom.svg(
         {
           className: "grid-outline",
           width: "100%",
--- a/devtools/client/inspector/grids/grid-inspector.js
+++ b/devtools/client/inspector/grids/grid-inspector.js
@@ -46,16 +46,17 @@ function GridInspector(inspector, window
   this.updateGridPanel = this.updateGridPanel.bind(this);
 
   this.onGridLayoutChange = this.onGridLayoutChange.bind(this);
   this.onHighlighterChange = this.onHighlighterChange.bind(this);
   this.onSetGridOverlayColor = this.onSetGridOverlayColor.bind(this);
   this.onShowBoxModelHighlighterForNode =
     this.onShowBoxModelHighlighterForNode.bind(this);
   this.onShowGridAreaHighlight = this.onShowGridAreaHighlight.bind(this);
+  this.onShowGridCellHighlight = this.onShowGridCellHighlight.bind(this);
   this.onSidebarSelect = this.onSidebarSelect.bind(this);
   this.onToggleGridHighlighter = this.onToggleGridHighlighter.bind(this);
   this.onToggleShowGridLineNumbers = this.onToggleShowGridLineNumbers.bind(this);
   this.onToggleShowInfiniteLines = this.onToggleShowInfiniteLines.bind(this);
 
   this.init();
 }
 
@@ -113,16 +114,17 @@ GridInspector.prototype = {
 
   getComponentProps() {
     return {
       getSwatchColorPickerTooltip: this.getSwatchColorPickerTooltip,
       setSelectedNode: this.setSelectedNode,
       onSetGridOverlayColor: this.onSetGridOverlayColor,
       onShowBoxModelHighlighterForNode: this.onShowBoxModelHighlighterForNode,
       onShowGridAreaHighlight: this.onShowGridAreaHighlight,
+      onShowGridCellHighlight: this.onShowGridCellHighlight,
       onToggleGridHighlighter: this.onToggleGridHighlighter,
       onToggleShowGridLineNumbers: this.onToggleShowGridLineNumbers,
       onToggleShowInfiniteLines: this.onToggleShowInfiniteLines,
     };
   },
 
   /**
    * Returns the initial color linked to a grid container. Will attempt to check the
@@ -351,16 +353,39 @@ GridInspector.prototype = {
 
     highlighterSettings.showGridArea = gridAreaName;
     highlighterSettings.color = color;
 
     this.highlighters.showGridHighlighter(node, highlighterSettings);
   },
 
   /**
+   * Highlights the grid cell in the CSS Grid Highlighter for the given grid.
+   *
+   * @param  {NodeFront} node
+   *         The NodeFront of the grid container element for which the grid
+   *         highlighter is highlighted for.
+   * @param  {Number|null} gridFragmentIndex
+   *         The index of the grid fragment for which the grid highlighter
+   *         is highlighted for.
+   * @param  {Number|null} rowNumber
+   *         The row number of the grid cell for which the grid highlighter
+   *         is highlighted for.
+   * @param  {Number|null} columnNumber
+   *         The column number of the grid cell for which the grid highlighter
+   *         is highlighted for.
+   */
+  onShowGridCellHighlight(node, gridFragmentIndex, rowNumber, columnNumber) {
+    let { highlighterSettings } = this.store.getState();
+    highlighterSettings.showGridCell = { gridFragmentIndex, rowNumber, columnNumber };
+
+    this.highlighters.showGridHighlighter(node, highlighterSettings);
+  },
+
+  /**
    * Handler for the inspector sidebar select event. Starts listening for
    * "grid-layout-changed" if the layout panel is visible. Otherwise, stop
    * listening for grid layout changes. Finally, refresh the layout view if
    * it is visible.
    */
   onSidebarSelect() {
     if (!this.isPanelVisible()) {
       this.layoutInspector.off("grid-layout-changed", this.onGridLayoutChange);
--- a/devtools/client/inspector/layout/layout.js
+++ b/devtools/client/inspector/layout/layout.js
@@ -39,16 +39,17 @@ LayoutView.prototype = {
     } = this.inspector.boxmodel.getComponentProps();
 
     let {
       getSwatchColorPickerTooltip,
       setSelectedNode,
       onSetGridOverlayColor,
       onShowBoxModelHighlighterForNode,
       onShowGridAreaHighlight,
+      onShowGridCellHighlight,
       onToggleGridHighlighter,
       onToggleShowGridLineNumbers,
       onToggleShowInfiniteLines,
     } = this.inspector.gridInspector.getComponentProps();
 
     let app = App({
       getSwatchColorPickerTooltip,
       setSelectedNode,
@@ -65,16 +66,17 @@ LayoutView.prototype = {
       showGridOutline: Services.prefs.getBoolPref(SHOW_GRID_OUTLINE_PREF),
 
       onHideBoxModelHighlighter,
       onSetGridOverlayColor,
       onShowBoxModelEditor,
       onShowBoxModelHighlighter,
       onShowBoxModelHighlighterForNode,
       onShowGridAreaHighlight,
+      onShowGridCellHighlight,
       onToggleGridHighlighter,
       onToggleShowGridLineNumbers,
       onToggleShowInfiniteLines,
     });
 
     let provider = createElement(Provider, {
       store: this.store,
       id: "layoutview",
--- a/devtools/server/actors/highlighters/css-grid.js
+++ b/devtools/server/actors/highlighters/css-grid.js
@@ -523,16 +523,17 @@ CssGridHighlighter.prototype = extend(Au
     root.setAttribute("style", "display: none");
     this.currentNode.offsetWidth;
 
     let { width, height } = getWindowDimensions(this.win);
 
     // Clear the canvas the grid area highlights.
     this.clearCanvas(width, height);
     this.clearGridAreas();
+    this.clearGridCell();
 
     // Start drawing the grid fragments.
     for (let i = 0; i < this.gridData.length; i++) {
       let fragment = this.gridData[i];
       let quad = this.currentQuads.content[i];
       this.renderFragment(fragment, quad);
     }
 
@@ -958,16 +959,17 @@ CssGridHighlighter.prototype = extend(Au
    *         Index of the grid fragment to render the grid cell highlight.
    * @param  {Number} rowNumber
    *         Row number of the grid cell to highlight.
    * @param  {Number} columnNumber
    *         Column number of the grid cell to highlight.
    */
   renderGridCell(gridFragmentIndex, rowNumber, columnNumber) {
     let fragment = this.gridData[gridFragmentIndex];
+
     if (!fragment) {
       return;
     }
 
     let row = fragment.rows.tracks[rowNumber - 1];
     let column = fragment.cols.tracks[columnNumber - 1];
 
     if (!row || !column) {
--- a/devtools/server/actors/utils/audionodes.json
+++ b/devtools/server/actors/utils/audionodes.json
@@ -104,10 +104,18 @@
     "properties": {
       "stream": { "MediaStream": true }
     }
   },
   "StereoPannerNode": {
     "properties": {
       "pan": { "param": true }
     }
+  },
+  "ConstantSourceNode": {
+    "source": true,
+    "properties": {
+      "offset": { "param": true }
+    }
+  },
+  "IIRFilterNode": {
   }
 }
--- a/devtools/shared/specs/webaudio.js
+++ b/devtools/shared/specs/webaudio.js
@@ -9,19 +9,20 @@ const {
   RetVal,
   generateActorSpec,
   types,
 } = require("devtools/shared/protocol");
 
 exports.NODE_CREATION_METHODS = [
   "createBufferSource", "createMediaElementSource", "createMediaStreamSource",
   "createMediaStreamDestination", "createScriptProcessor", "createAnalyser",
-  "createGain", "createDelay", "createBiquadFilter", "createWaveShaper",
-  "createPanner", "createConvolver", "createChannelSplitter", "createChannelMerger",
-  "createDynamicsCompressor", "createOscillator", "createStereoPanner"
+  "createGain", "createDelay", "createBiquadFilter", "createIIRFilter",
+  "createWaveShaper", "createPanner", "createConvolver", "createChannelSplitter",
+  "createChannelMerger", "createDynamicsCompressor", "createOscillator",
+  "createStereoPanner", "createConstantSource"
 ];
 
 exports.AUTOMATION_METHODS = [
   "setValueAtTime", "linearRampToValueAtTime", "exponentialRampToValueAtTime",
   "setTargetAtTime", "setValueCurveAtTime", "cancelScheduledValues"
 ];
 
 exports.NODE_ROUTING_METHODS = [
--- a/docshell/base/nsDefaultURIFixup.cpp
+++ b/docshell/base/nsDefaultURIFixup.cpp
@@ -15,17 +15,17 @@
 #ifdef MOZ_TOOLKIT_SEARCH
 #include "nsIBrowserSearchService.h"
 #endif
 
 #include "nsIURIFixup.h"
 #include "nsDefaultURIFixup.h"
 #include "mozilla/Preferences.h"
 #include "mozilla/dom/ContentChild.h"
-#include "mozilla/ipc/InputStreamUtils.h"
+#include "mozilla/ipc/IPCStreamUtils.h"
 #include "mozilla/ipc/URIUtils.h"
 #include "mozilla/Tokenizer.h"
 #include "nsIObserverService.h"
 #include "nsXULAppAPI.h"
 
 // Used to check if external protocol schemes are usable
 #include "nsCExternalHandlerService.h"
 #include "nsIExternalProtocolService.h"
@@ -427,33 +427,30 @@ nsDefaultURIFixup::KeywordToURI(const ns
   keyword.Trim(" ");
 
   if (XRE_IsContentProcess()) {
     dom::ContentChild* contentChild = dom::ContentChild::GetSingleton();
     if (!contentChild) {
       return NS_ERROR_NOT_AVAILABLE;
     }
 
-    ipc::OptionalInputStreamParams postData;
+    ipc::OptionalIPCStream postData;
     ipc::OptionalURIParams uri;
     nsAutoString providerName;
     if (!contentChild->SendKeywordToURI(keyword, &providerName, &postData,
                                         &uri)) {
       return NS_ERROR_FAILURE;
     }
 
     CopyUTF8toUTF16(keyword, info->mKeywordAsSent);
     info->mKeywordProviderName = providerName;
 
     if (aPostData) {
-      nsTArray<ipc::FileDescriptor> fds;
-      nsCOMPtr<nsIInputStream> temp = DeserializeInputStream(postData, fds);
+      nsCOMPtr<nsIInputStream> temp = ipc::DeserializeIPCStream(postData);
       temp.forget(aPostData);
-
-      MOZ_ASSERT(fds.IsEmpty());
     }
 
     nsCOMPtr<nsIURI> temp = DeserializeURI(uri);
     info->mPreferredURI = temp.forget();
     return NS_OK;
   }
 
 #ifdef MOZ_TOOLKIT_SEARCH
--- a/dom/ipc/ContentParent.cpp
+++ b/dom/ipc/ContentParent.cpp
@@ -66,17 +66,17 @@
 #include "mozilla/gfx/gfxVars.h"
 #include "mozilla/gfx/GPUProcessManager.h"
 #include "mozilla/hal_sandbox/PHalParent.h"
 #include "mozilla/ipc/BackgroundChild.h"
 #include "mozilla/ipc/BackgroundParent.h"
 #include "mozilla/ipc/FileDescriptorUtils.h"
 #include "mozilla/ipc/PSendStreamParent.h"
 #include "mozilla/ipc/TestShellParent.h"
-#include "mozilla/ipc/InputStreamUtils.h"
+#include "mozilla/ipc/IPCStreamUtils.h"
 #include "mozilla/jsipc/CrossProcessObjectWrappers.h"
 #include "mozilla/layers/PAPZParent.h"
 #include "mozilla/layers/CompositorThread.h"
 #include "mozilla/layers/ImageBridgeParent.h"
 #include "mozilla/layers/LayerTreeOwnerTracker.h"
 #include "mozilla/layout/RenderFrameParent.h"
 #include "mozilla/LookAndFeel.h"
 #include "mozilla/media/MediaParent.h"
@@ -3898,17 +3898,17 @@ ContentParent::SendPBrowserConstructor(P
                                                  aChromeFlags,
                                                  aCpId,
                                                  aIsForBrowser);
 }
 
 mozilla::ipc::IPCResult
 ContentParent::RecvKeywordToURI(const nsCString& aKeyword,
                                 nsString* aProviderName,
-                                OptionalInputStreamParams* aPostData,
+                                OptionalIPCStream* aPostData,
                                 OptionalURIParams* aURI)
 {
   *aPostData = void_t();
   *aURI = void_t();
 
   nsCOMPtr<nsIURIFixup> fixup = do_GetService(NS_URIFIXUP_CONTRACTID);
   if (!fixup) {
     return IPC_OK();
@@ -3918,19 +3918,19 @@ ContentParent::RecvKeywordToURI(const ns
   nsCOMPtr<nsIURIFixupInfo> info;
 
   if (NS_FAILED(fixup->KeywordToURI(aKeyword, getter_AddRefs(postData),
                                     getter_AddRefs(info)))) {
     return IPC_OK();
   }
   info->GetKeywordProviderName(*aProviderName);
 
-  nsTArray<mozilla::ipc::FileDescriptor> fds;
-  SerializeInputStream(postData, *aPostData, fds);
-  MOZ_ASSERT(fds.IsEmpty());
+  AutoIPCStream autoStream;
+  autoStream.Serialize(postData, this);
+  *aPostData = autoStream.TakeOptionalValue();
 
   nsCOMPtr<nsIURI> uri;
   info->GetPreferredURI(getter_AddRefs(uri));
   SerializeURI(uri, *aURI);
   return IPC_OK();
 }
 
 mozilla::ipc::IPCResult
--- a/dom/ipc/ContentParent.h
+++ b/dom/ipc/ContentParent.h
@@ -991,17 +991,17 @@ private:
                                                                       const bool& aHidden) override;
 
   virtual mozilla::ipc::IPCResult RecvAudioChannelServiceStatus(const bool& aTelephonyChannel,
                                                                 const bool& aContentOrNormalChannel,
                                                                 const bool& aAnyChannel) override;
 
   virtual mozilla::ipc::IPCResult RecvKeywordToURI(const nsCString& aKeyword,
                                                    nsString* aProviderName,
-                                                   OptionalInputStreamParams* aPostData,
+                                                   OptionalIPCStream* aPostData,
                                                    OptionalURIParams* aURI) override;
 
   virtual mozilla::ipc::IPCResult RecvNotifyKeywordSearchLoading(const nsString &aProvider,
                                                                  const nsString &aKeyword) override;
 
   virtual mozilla::ipc::IPCResult RecvCopyFavicon(const URIParams& aOldURI,
                                                   const URIParams& aNewURI,
                                                   const IPC::Principal& aLoadingPrincipal,
--- a/dom/ipc/PContent.ipdl
+++ b/dom/ipc/PContent.ipdl
@@ -39,17 +39,17 @@ include protocol PWebBrowserPersistDocum
 include protocol PWebrtcGlobal;
 include protocol PPresentation;
 include protocol PURLClassifier;
 include protocol PVRManager;
 include protocol PVideoDecoderManager;
 include protocol PFlyWebPublishedServer;
 include DOMTypes;
 include JavaScriptTypes;
-include InputStreamParams;
+include IPCStream;
 include PTabContext;
 include URIParams;
 include PluginTypes;
 include ProtocolTypes;
 include PBackgroundSharedTypes;
 include PContentPermission;
 include ServiceWorkerConfiguration;
 include GraphicsMessages;
@@ -812,17 +812,17 @@ parent:
 
     async AudioChannelServiceStatus(bool aActiveTelephonyChannel,
                                     bool aContentOrNormalChannel,
                                     bool aAnyActiveChannel);
 
     async AudioChannelChangeDefVolChannel(int32_t aChannel, bool aHidden);
 
     sync KeywordToURI(nsCString keyword)
-        returns (nsString providerName, OptionalInputStreamParams postData, OptionalURIParams uri);
+        returns (nsString providerName, OptionalIPCStream postData, OptionalURIParams uri);
 
     sync NotifyKeywordSearchLoading(nsString providerName, nsString keyword);
 
     async CopyFavicon(URIParams oldURI, URIParams newURI, Principal aLoadingPrincipal, bool isPrivate);
 
     // Tell the compositor to allocate a layer tree id for nested remote mozbrowsers.
     sync AllocateLayerTreeId(ContentParentId cpId, TabId tabId)
         returns (uint64_t id);
--- a/dom/media/CubebUtils.cpp
+++ b/dom/media/CubebUtils.cpp
@@ -49,16 +49,17 @@ LazyLogModule gCubebLog("cubeb");
 
 void CubebLogCallback(const char* aFmt, ...)
 {
   char buffer[256];
 
   va_list arglist;
   va_start(arglist, aFmt);
   VsprintfLiteral (buffer, aFmt, arglist);
+  printf("%s\n", buffer);
   MOZ_LOG(gCubebLog, LogLevel::Error, ("%s", buffer));
   va_end(arglist);
 }
 
 // This mutex protects the variables below.
 StaticMutex sMutex;
 enum class CubebState {
   Uninitialized = 0,
@@ -328,18 +329,20 @@ cubeb* GetCubebContextUnlocked()
       sBrandName, "Did not initialize sbrandName, and not on the main thread?");
   }
 
   int rv = cubeb_init(&sCubebContext, sBrandName);
   NS_WARNING_ASSERTION(rv == CUBEB_OK, "Could not get a cubeb context.");
   sCubebState = (rv == CUBEB_OK) ? CubebState::Initialized : CubebState::Uninitialized;
 
   if (MOZ_LOG_TEST(gCubebLog, LogLevel::Verbose)) {
+    printf("Setting callback, verbose\n");
     cubeb_set_log_callback(CUBEB_LOG_VERBOSE, CubebLogCallback);
   } else if (MOZ_LOG_TEST(gCubebLog, LogLevel::Error)) {
+    printf("Setting callback, error\n");
     cubeb_set_log_callback(CUBEB_LOG_NORMAL, CubebLogCallback);
   }
 
   return sCubebContext;
 }
 
 void ReportCubebBackendUsed()
 {
--- a/dom/network/PUDPSocket.ipdl
+++ b/dom/network/PUDPSocket.ipdl
@@ -2,18 +2,21 @@
 /* vim: set sw=2 ts=8 et tw=80 ft=cpp : */
 
 /* 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 protocol PNecko;
 include protocol PBackground;
-include protocol PBlob;
-include InputStreamParams;
+include protocol PBlob; //FIXME: bug #792908
+include protocol PFileDescriptorSet; // FIXME: bug #792908
+include protocol PSendStream; //FIXME: bug #792908
+
+include IPCStream;
 
 include "mozilla/net/NeckoMessageUtils.h";
 include "mozilla/net/DNS.h";
 include "prio.h";
 
 using mozilla::net::NetAddr from "mozilla/net/DNS.h";
 using struct mozilla::void_t from "ipc/IPCMessageUtils.h";
 
@@ -24,17 +27,17 @@ struct UDPAddressInfo {
 
 union UDPSocketAddr {
   UDPAddressInfo;
   NetAddr;
 };
 
 union UDPData {
   uint8_t[];
-  InputStreamParams;
+  IPCStream;
 };
 
 namespace mozilla {
 namespace net {
 
 //-------------------------------------------------------------------
 protocol PUDPSocket
 {
--- a/dom/network/UDPSocketChild.cpp
+++ b/dom/network/UDPSocketChild.cpp
@@ -2,18 +2,19 @@
 /* vim: set ts=8 sts=2 et sw=2 tw=80: */
 /* This Source Code Form is subject to the terms of the Mozilla Public
  * License, v. 2.0. If a copy of the MPL was not distributed with this file,
  * You can obtain one at http://mozilla.org/MPL/2.0/. */
 
 #include "UDPSocketChild.h"
 #include "mozilla/SizePrintfMacros.h"
 #include "mozilla/Unused.h"
-#include "mozilla/ipc/InputStreamUtils.h"
+#include "mozilla/ipc/IPCStreamUtils.h"
 #include "mozilla/net/NeckoChild.h"
+#include "mozilla/dom/ContentChild.h"
 #include "mozilla/dom/PermissionMessageUtils.h"
 #include "mozilla/ipc/BackgroundChild.h"
 #include "mozilla/ipc/PBackgroundChild.h"
 #include "mozilla/ipc/BackgroundUtils.h"
 #include "mozilla/ipc/PBackgroundSharedTypes.h"
 #include "nsIIPCBackgroundChildCreateCallback.h"
 
 using mozilla::net::gNeckoChild;
@@ -272,24 +273,23 @@ UDPSocketChild::SendDataInternal(const U
 
 NS_IMETHODIMP
 UDPSocketChild::SendBinaryStream(const nsACString& aHost,
                                  uint16_t aPort,
                                  nsIInputStream* aStream)
 {
   NS_ENSURE_ARG(aStream);
 
-  OptionalInputStreamParams stream;
-  nsTArray<mozilla::ipc::FileDescriptor> fds;
-  SerializeInputStream(aStream, stream, fds);
-
-  MOZ_ASSERT(fds.IsEmpty());
+  mozilla::ipc::AutoIPCStream autoStream;
+  autoStream.Serialize(aStream,
+                       static_cast<mozilla::dom::ContentChild*>(gNeckoChild->Manager()));
 
   UDPSOCKET_LOG(("%s: %s:%u", __FUNCTION__, PromiseFlatCString(aHost).get(), aPort));
-  SendOutgoingData(UDPData(stream), UDPSocketAddr(UDPAddressInfo(nsCString(aHost), aPort)));
+  SendOutgoingData(UDPData(autoStream.TakeOptionalValue()),
+                   UDPSocketAddr(UDPAddressInfo(nsCString(aHost), aPort)));
 
   return NS_OK;
 }
 
 NS_IMETHODIMP
 UDPSocketChild::JoinMulticast(const nsACString& aMulticastAddress,
                               const nsACString& aInterface)
 {
--- a/dom/network/UDPSocketParent.cpp
+++ b/dom/network/UDPSocketParent.cpp
@@ -366,18 +366,18 @@ UDPSocketParent::RecvOutgoingData(const 
       return IPC_FAIL(this, "Content tried to send non STUN packet");
     }
   }
 
   switch(aData.type()) {
     case UDPData::TArrayOfuint8_t:
       Send(aData.get_ArrayOfuint8_t(), aAddr);
       break;
-    case UDPData::TInputStreamParams:
-      Send(aData.get_InputStreamParams(), aAddr);
+    case UDPData::TIPCStream:
+      Send(aData.get_IPCStream(), aAddr);
       break;
     default:
       MOZ_ASSERT(false, "Invalid data type!");
       return IPC_OK();
   }
 
   return IPC_OK();
 }
@@ -407,21 +407,20 @@ UDPSocketParent::Send(const InfallibleTA
   }
 
   if (NS_WARN_IF(NS_FAILED(rv)) || count == 0) {
     FireInternalError(__LINE__);
   }
 }
 
 void
-UDPSocketParent::Send(const InputStreamParams& aStream,
+UDPSocketParent::Send(const IPCStream& aStream,
                       const UDPSocketAddr& aAddr)
 {
-  nsTArray<mozilla::ipc::FileDescriptor> fds;
-  nsCOMPtr<nsIInputStream> stream = DeserializeInputStream(aStream, fds);
+  nsCOMPtr<nsIInputStream> stream = DeserializeIPCStream(aStream);
 
   if (NS_WARN_IF(!stream)) {
     return;
   }
 
   nsresult rv;
   switch(aAddr.type()) {
     case UDPSocketAddr::TUDPAddressInfo: {
--- a/dom/network/UDPSocketParent.h
+++ b/dom/network/UDPSocketParent.h
@@ -53,17 +53,17 @@ public:
   virtual mozilla::ipc::IPCResult RecvLeaveMulticast(const nsCString& aMulticastAddress,
                                                      const nsCString& aInterface) override;
 
 private:
   virtual ~UDPSocketParent();
 
   virtual void ActorDestroy(ActorDestroyReason why) override;
   void Send(const InfallibleTArray<uint8_t>& aData, const UDPSocketAddr& aAddr);
-  void Send(const InputStreamParams& aStream, const UDPSocketAddr& aAddr);
+  void Send(const IPCStream& aStream, const UDPSocketAddr& aAddr);
   nsresult BindInternal(const nsCString& aHost, const uint16_t& aPort,
                         const bool& aAddressReuse, const bool& aLoopback,
                         const uint32_t& recvBufferSize,
                         const uint32_t& sendBufferSize);
   nsresult ConnectInternal(const nsCString& aHost, const uint16_t& aPort);
   void FireInternalError(uint32_t aLineNo);
   void SendInternalError(nsIEventTarget *aThread,
                          uint32_t aLineNo);
--- a/dom/push/PushNotifier.cpp
+++ b/dom/push/PushNotifier.cpp
@@ -103,17 +103,21 @@ PushNotifier::Dispatch(PushDispatcher& a
     Unused << NS_WARN_IF(NS_FAILED(aDispatcher.NotifyObservers()));
 
     nsTArray<ContentParent*> contentActors;
     ContentParent::GetAll(contentActors);
     if (!contentActors.IsEmpty()) {
       // At least one content process is active, so e10s must be enabled.
       // Broadcast a message to notify observers and service workers.
       for (uint32_t i = 0; i < contentActors.Length(); ++i) {
-        Unused << NS_WARN_IF(!aDispatcher.SendToChild(contentActors[i]));
+        if (aDispatcher.SendToChild(contentActors[i])) {
+          // Only send the push message to the first content process to avoid
+          // multiple SWs showing the same notification. See bug 1300112.
+          break;
+        }
       }
       return NS_OK;
     }
 
     if (BrowserTabsRemoteAutostart()) {
       // e10s is enabled, but no content processes are active.
       return aDispatcher.HandleNoChildProcesses();
     }
--- a/dom/quota/ActorsParent.cpp
+++ b/dom/quota/ActorsParent.cpp
@@ -482,31 +482,36 @@ namespace {
 class OriginInfo final
 {
   friend class GroupInfo;
   friend class QuotaManager;
   friend class QuotaObject;
 
 public:
   OriginInfo(GroupInfo* aGroupInfo, const nsACString& aOrigin,
-             uint64_t aUsage, int64_t aAccessTime)
-  : mGroupInfo(aGroupInfo), mOrigin(aOrigin), mUsage(aUsage),
-    mAccessTime(aAccessTime)
-  {
-    MOZ_COUNT_CTOR(OriginInfo);
-  }
+             uint64_t aUsage, int64_t aAccessTime, bool aPersisted);
 
   NS_INLINE_DECL_THREADSAFE_REFCOUNTING(OriginInfo)
 
   int64_t
-  AccessTime() const
+  LockedAccessTime() const
   {
+    AssertCurrentThreadOwnsQuotaMutex();
+
     return mAccessTime;
   }
 
+  bool
+  LockedPersisted() const
+  {
+    AssertCurrentThreadOwnsQuotaMutex();
+
+    return mPersisted;
+  }
+
 private:
   // Private destructor, to discourage deletion outside of Release():
   ~OriginInfo()
   {
     MOZ_COUNT_DTOR(OriginInfo);
 
     MOZ_ASSERT(!mQuotaObjects.Count());
   }
@@ -517,38 +522,44 @@ private:
   void
   LockedUpdateAccessTime(int64_t aAccessTime)
   {
     AssertCurrentThreadOwnsQuotaMutex();
 
     mAccessTime = aAccessTime;
   }
 
+  void
+  LockedPersist();
+
   nsDataHashtable<nsStringHashKey, QuotaObject*> mQuotaObjects;
 
   GroupInfo* mGroupInfo;
   const nsCString mOrigin;
   uint64_t mUsage;
   int64_t mAccessTime;
+  bool mPersisted;
 };
 
 class OriginInfoLRUComparator
 {
 public:
   bool
   Equals(const OriginInfo* a, const OriginInfo* b) const
   {
-    return
-      a && b ? a->AccessTime() == b->AccessTime() : !a && !b ? true : false;
+    return a && b ?
+             a->LockedAccessTime() == b->LockedAccessTime() :
+             !a && !b ? true : false;
   }
 
   bool
   LessThan(const OriginInfo* a, const OriginInfo* b) const
   {
-    return a && b ? a->AccessTime() < b->AccessTime() : b ? true : false;
+    return
+      a && b ? a->LockedAccessTime() < b->LockedAccessTime() : b ? true : false;
   }
 };
 
 class GroupInfo final
 {
   friend class GroupInfoPair;
   friend class OriginInfo;
   friend class QuotaManager;
@@ -2712,18 +2723,20 @@ QuotaObject::MaybeUpdateSize(int64_t aSi
 
   if (mSize > aSize) {
     if (aTruncate) {
       const int64_t delta = mSize - aSize;
 
       AssertNoUnderflow(quotaManager->mTemporaryStorageUsage, delta);
       quotaManager->mTemporaryStorageUsage -= delta;
 
-      AssertNoUnderflow(groupInfo->mUsage, delta);
-      groupInfo->mUsage -= delta;
+      if (!mOriginInfo->LockedPersisted()) {
+        AssertNoUnderflow(groupInfo->mUsage, delta);
+        groupInfo->mUsage -= delta;
+      }
 
       AssertNoUnderflow(mOriginInfo->mUsage, delta);
       mOriginInfo->mUsage -= delta;
 
       mSize = aSize;
     }
     return true;
   }
@@ -2737,45 +2750,50 @@ QuotaObject::MaybeUpdateSize(int64_t aSi
   uint64_t delta = aSize - mSize;
 
   AssertNoOverflow(mOriginInfo->mUsage, delta);
   uint64_t newUsage = mOriginInfo->mUsage + delta;
 
   // Temporary storage has no limit for origin usage (there's a group and the
   // global limit though).
 
-  AssertNoOverflow(groupInfo->mUsage, delta);
-  uint64_t newGroupUsage = groupInfo->mUsage + delta;
-
-  uint64_t groupUsage = groupInfo->mUsage;
-  if (complementaryGroupInfo) {
-    AssertNoOverflow(groupUsage, complementaryGroupInfo->mUsage);
-    groupUsage += complementaryGroupInfo->mUsage;
-  }
-
-  // Temporary storage has a hard limit for group usage (20 % of the global
-  // limit).
-  AssertNoOverflow(groupUsage, delta);
-  if (groupUsage + delta > quotaManager->GetGroupLimit()) {
-    return false;
+  uint64_t newGroupUsage = groupInfo->mUsage;
+  if (!mOriginInfo->LockedPersisted()) {
+    AssertNoOverflow(groupInfo->mUsage, delta);
+    newGroupUsage += delta;
+
+    uint64_t groupUsage = groupInfo->mUsage;
+    if (complementaryGroupInfo) {
+      AssertNoOverflow(groupUsage, complementaryGroupInfo->mUsage);
+      groupUsage += complementaryGroupInfo->mUsage;
+    }
+
+    // Temporary storage has a hard limit for group usage (20 % of the global
+    // limit).
+    AssertNoOverflow(groupUsage, delta);
+    if (groupUsage + delta > quotaManager->GetGroupLimit()) {
+      return false;
+    }
   }
 
   AssertNoOverflow(quotaManager->mTemporaryStorageUsage, delta);
   uint64_t newTemporaryStorageUsage = quotaManager->mTemporaryStorageUsage +
                                       delta;
 
   if (newTemporaryStorageUsage > quotaManager->mTemporaryStorageLimit) {
     // This will block the thread without holding the lock while waitting.
 
     AutoTArray<RefPtr<DirectoryLockImpl>, 10> locks;
 
     uint64_t sizeToBeFreed =
       quotaManager->LockedCollectOriginsForEviction(delta, locks);
 
     if (!sizeToBeFreed) {
+      // XXX prompt for asking to delete persistent origins if there is any
+      // persistent origin.
       return false;
     }
 
     NS_ASSERTION(sizeToBeFreed >= delta, "Huh?");
 
     {
       MutexAutoUnlock autoUnlock(quotaManager->mQuotaMutex);
 
@@ -2811,48 +2829,53 @@ QuotaObject::MaybeUpdateSize(int64_t aSi
     // essential variables and recheck the group limit.
 
     AssertNoUnderflow(aSize, mSize);
     delta = aSize - mSize;
 
     AssertNoOverflow(mOriginInfo->mUsage, delta);
     newUsage = mOriginInfo->mUsage + delta;
 
-    AssertNoOverflow(groupInfo->mUsage, delta);
-    newGroupUsage = groupInfo->mUsage + delta;
-
-    groupUsage = groupInfo->mUsage;
-    if (complementaryGroupInfo) {
-      AssertNoOverflow(groupUsage, complementaryGroupInfo->mUsage);
-      groupUsage += complementaryGroupInfo->mUsage;
-    }
-
-    AssertNoOverflow(groupUsage, delta);
-    if (groupUsage + delta > quotaManager->GetGroupLimit()) {
-      // Unfortunately some other thread increased the group usage in the
-      // meantime and we are not below the group limit anymore.
-
-      // However, the origin eviction must be finalized in this case too.
-      MutexAutoUnlock autoUnlock(quotaManager->mQuotaMutex);
-
-      quotaManager->FinalizeOriginEviction(locks);
-
-      return false;
+    newGroupUsage = groupInfo->mUsage;
+    if (!mOriginInfo->LockedPersisted()) {
+      AssertNoOverflow(groupInfo->mUsage, delta);
+      newGroupUsage += delta;
+
+      uint64_t groupUsage = groupInfo->mUsage;
+      if (complementaryGroupInfo) {
+        AssertNoOverflow(groupUsage, complementaryGroupInfo->mUsage);
+        groupUsage += complementaryGroupInfo->mUsage;
+      }
+
+      AssertNoOverflow(groupUsage, delta);
+      if (groupUsage + delta > quotaManager->GetGroupLimit()) {
+        // Unfortunately some other thread increased the group usage in the
+        // meantime and we are not below the group limit anymore.
+
+        // However, the origin eviction must be finalized in this case too.
+        MutexAutoUnlock autoUnlock(quotaManager->mQuotaMutex);
+
+        quotaManager->FinalizeOriginEviction(locks);
+
+        return false;
+      }
     }
 
     AssertNoOverflow(quotaManager->mTemporaryStorageUsage, delta);
     newTemporaryStorageUsage = quotaManager->mTemporaryStorageUsage + delta;
 
     NS_ASSERTION(newTemporaryStorageUsage <=
                  quotaManager->mTemporaryStorageLimit, "How come?!");
 
     // Ok, we successfully freed enough space and the operation can continue
     // without throwing the quota error.
     mOriginInfo->mUsage = newUsage;
-    groupInfo->mUsage = newGroupUsage;
+    if (!mOriginInfo->LockedPersisted()) {
+      groupInfo->mUsage = newGroupUsage;
+    }
     quotaManager->mTemporaryStorageUsage = newTemporaryStorageUsage;;
 
     // Some other thread could increase the size in the meantime, but no more
     // than this one.
     MOZ_ASSERT(mSize < aSize);
     mSize = aSize;
 
     // Finally, release IO thread only objects and allow next synchronized
@@ -2860,17 +2883,19 @@ QuotaObject::MaybeUpdateSize(int64_t aSi
     MutexAutoUnlock autoUnlock(quotaManager->mQuotaMutex);
 
     quotaManager->FinalizeOriginEviction(locks);
 
     return true;
   }
 
   mOriginInfo->mUsage = newUsage;
-  groupInfo->mUsage = newGroupUsage;
+  if (!mOriginInfo->LockedPersisted()) {
+    groupInfo->mUsage = newGroupUsage;
+  }
   quotaManager->mTemporaryStorageUsage = newTemporaryStorageUsage;
 
   mSize = aSize;
 
   return true;
 }
 
 void
@@ -3136,16 +3161,20 @@ QuotaManager::CollectOriginsForEviction(
     GetInactiveOriginInfos(nsTArray<RefPtr<OriginInfo>>& aOriginInfos,
                            nsTArray<DirectoryLockImpl*>& aLocks,
                            nsTArray<OriginInfo*>& aInactiveOriginInfos)
     {
       for (OriginInfo* originInfo : aOriginInfos) {
         MOZ_ASSERT(originInfo->mGroupInfo->mPersistenceType !=
                      PERSISTENCE_TYPE_PERSISTENT);
 
+        if (originInfo->LockedPersisted()) {
+          continue;
+        }
+
         OriginScope originScope = OriginScope::FromOrigin(originInfo->mOrigin);
 
         bool match = false;
         for (uint32_t j = aLocks.Length(); j > 0; j--) {
           DirectoryLockImpl* lock = aLocks[j - 1];
           if (originScope.Matches(lock->GetOriginScope())) {
             match = true;
             break;
@@ -3388,17 +3417,18 @@ QuotaManager::Shutdown()
   }
 }
 
 void
 QuotaManager::InitQuotaForOrigin(PersistenceType aPersistenceType,
                                  const nsACString& aGroup,
                                  const nsACString& aOrigin,
                                  uint64_t aUsageBytes,
-                                 int64_t aAccessTime)
+                                 int64_t aAccessTime,
+                                 bool aPersisted)
 {
   AssertIsOnIOThread();
   MOZ_ASSERT(aPersistenceType != PERSISTENCE_TYPE_PERSISTENT);
 
   MutexAutoLock lock(mQuotaMutex);
 
   GroupInfoPair* pair;
   if (!mGroupInfoPairs.Get(aGroup, &pair)) {
@@ -3409,17 +3439,17 @@ QuotaManager::InitQuotaForOrigin(Persist
 
   RefPtr<GroupInfo> groupInfo = pair->LockedGetGroupInfo(aPersistenceType);
   if (!groupInfo) {
     groupInfo = new GroupInfo(pair, aPersistenceType, aGroup);
     pair->LockedSetGroupInfo(aPersistenceType, groupInfo);
   }
 
   RefPtr<OriginInfo> originInfo =
-    new OriginInfo(groupInfo, aOrigin, aUsageBytes, aAccessTime);
+    new OriginInfo(groupInfo, aOrigin, aUsageBytes, aAccessTime, aPersisted);
   groupInfo->LockedAddOriginInfo(originInfo);
 }
 
 void
 QuotaManager::DecreaseUsageForOrigin(PersistenceType aPersistenceType,
                                      const nsACString& aGroup,
                                      const nsACString& aOrigin,
                                      int64_t aSize)
@@ -3878,33 +3908,34 @@ QuotaManager::InitializeRepository(Persi
                                           suffix,
                                           group,
                                           origin);
     if (NS_WARN_IF(NS_FAILED(rv))) {
       return rv;
     }
 
     rv = InitializeOrigin(aPersistenceType, group, origin, timestamp,
-                          childDirectory);
+                          /* aPersisted */ false, childDirectory);
     if (NS_WARN_IF(NS_FAILED(rv))) {
       return rv;
     }
   }
   if (NS_WARN_IF(NS_FAILED(rv))) {
     return rv;
   }
 
   return NS_OK;
 }
 
 nsresult
 QuotaManager::InitializeOrigin(PersistenceType aPersistenceType,
                                const nsACString& aGroup,
                                const nsACString& aOrigin,
                                int64_t aAccessTime,
+                               bool aPersisted,
                                nsIFile* aDirectory)
 {
   AssertIsOnIOThread();
 
   nsresult rv;
 
   bool trackQuota = aPersistenceType != PERSISTENCE_TYPE_PERSISTENT;
 
@@ -3967,17 +3998,17 @@ QuotaManager::InitializeOrigin(Persisten
 
     rv = mClients[clientType]->InitOrigin(aPersistenceType, aGroup, aOrigin,
                                           usageInfo);
     NS_ENSURE_SUCCESS(rv, rv);
   }
 
   if (trackQuota) {
     InitQuotaForOrigin(aPersistenceType, aGroup, aOrigin,
-                       usageInfo->TotalUsage(), aAccessTime);
+                       usageInfo->TotalUsage(), aAccessTime, aPersisted);
   }
 
   return NS_OK;
 }
 
 nsresult
 QuotaManager::MaybeUpgradeIndexedDBDirectory()
 {
@@ -4852,17 +4883,17 @@ QuotaManager::EnsureOriginIsInitializedI
       if (NS_WARN_IF(NS_FAILED(rv))) {
         return rv;
       }
 
       MOZ_ASSERT(timestamp <= PR_Now());
     }
 
     rv = InitializeOrigin(aPersistenceType, aGroup, aOrigin, timestamp,
-                          directory);
+                          /* aPersisted */ true, directory);
     NS_ENSURE_SUCCESS(rv, rv);
 
     mInitializedOrigins.AppendElement(aOrigin);
   } else if (created) {
     timestamp = PR_Now();
 
     rv = CreateDirectoryMetadata(directory,
                                  timestamp,
@@ -4876,17 +4907,17 @@ QuotaManager::EnsureOriginIsInitializedI
     rv = CreateDirectoryMetadata2(directory,
                                   timestamp,
                                   aSuffix,
                                   aGroup,
                                   aOrigin);
     NS_ENSURE_SUCCESS(rv, rv);
 
     rv = InitializeOrigin(aPersistenceType, aGroup, aOrigin, timestamp,
-                          directory);
+                          /* aPersisted */ false, directory);
     NS_ENSURE_SUCCESS(rv, rv);
   }
 
   directory.forget(aDirectory);
   *aCreated = created;
   return NS_OK;
 }
 
@@ -5237,16 +5268,19 @@ QuotaManager::CheckTemporaryStorageLimit
           }
           if (defaultGroupInfo) {
             originInfos.AppendElements(defaultGroupInfo->mOriginInfos);
           }
           originInfos.Sort(OriginInfoLRUComparator());
 
           for (uint32_t i = 0; i < originInfos.Length(); i++) {
             OriginInfo* originInfo = originInfos[i];
+            if (originInfo->LockedPersisted()) {
+              continue;
+            }
 
             doomedOriginInfos.AppendElement(originInfo);
             groupUsage -= originInfo->mUsage;
 
             if (groupUsage <= quotaManager->GetGroupLimit()) {
               break;
             }
           }
@@ -5276,17 +5310,18 @@ QuotaManager::CheckTemporaryStorageLimit
 
         groupInfo = pair->LockedGetGroupInfo(PERSISTENCE_TYPE_DEFAULT);
         if (groupInfo) {
           originInfos.AppendElements(groupInfo->mOriginInfos);
         }
       }
 
       for (uint32_t index = originInfos.Length(); index > 0; index--) {
-        if (doomedOriginInfos.Contains(originInfos[index - 1])) {
+        if (doomedOriginInfos.Contains(originInfos[index - 1]) ||
+            originInfos[index - 1]->LockedPersisted()) {
           originInfos.RemoveElementAt(index - 1);
         }
       }
 
       originInfos.Sort(OriginInfoLRUComparator());
 
       for (uint32_t i = 0; i < originInfos.Length(); i++) {
         if (mTemporaryStorageUsage - usage <= mTemporaryStorageLimit) {
@@ -5299,16 +5334,23 @@ QuotaManager::CheckTemporaryStorageLimit
 
       doomedOriginInfos.AppendElements(originInfos);
     }
   }
 
   for (uint32_t index = 0; index < doomedOriginInfos.Length(); index++) {
     OriginInfo* doomedOriginInfo = doomedOriginInfos[index];
 
+#ifdef DEBUG
+    {
+      MutexAutoLock lock(mQuotaMutex);
+      MOZ_ASSERT(!doomedOriginInfo->LockedPersisted());
+    }
+#endif
+
     DeleteFilesForOrigin(doomedOriginInfo->mGroupInfo->mPersistenceType,
                          doomedOriginInfo->mOrigin);
   }
 
   nsTArray<OriginParams> doomedOrigins;
   {
     MutexAutoLock lock(mQuotaMutex);
 
@@ -5402,34 +5444,62 @@ QuotaManager::GetDirectoryLockTable(Pers
       MOZ_CRASH("Bad persistence type value!");
   }
 }
 
 /*******************************************************************************
  * Local class implementations
  ******************************************************************************/
 
+OriginInfo::OriginInfo(GroupInfo* aGroupInfo, const nsACString& aOrigin,
+                       uint64_t aUsage, int64_t aAccessTime, bool aPersisted)
+  : mGroupInfo(aGroupInfo), mOrigin(aOrigin), mUsage(aUsage),
+    mAccessTime(aAccessTime), mPersisted(aPersisted)
+{
+  MOZ_ASSERT(aGroupInfo);
+  MOZ_ASSERT_IF(aPersisted,
+                aGroupInfo->mPersistenceType == PERSISTENCE_TYPE_DEFAULT);
+
+  MOZ_COUNT_CTOR(OriginInfo);
+}
+
 void
 OriginInfo::LockedDecreaseUsage(int64_t aSize)
 {
   AssertCurrentThreadOwnsQuotaMutex();
 
   AssertNoUnderflow(mUsage, aSize);
   mUsage -= aSize;
 
-  AssertNoUnderflow(mGroupInfo->mUsage, aSize);
-  mGroupInfo->mUsage -= aSize;
+  if (!LockedPersisted()) {
+    AssertNoUnderflow(mGroupInfo->mUsage, aSize);
+    mGroupInfo->mUsage -= aSize;
+  }
 
   QuotaManager* quotaManager = QuotaManager::Get();
   MOZ_ASSERT(quotaManager);
 
   AssertNoUnderflow(quotaManager->mTemporaryStorageUsage, aSize);
   quotaManager->mTemporaryStorageUsage -= aSize;
 }
 
+void
+OriginInfo::LockedPersist()
+{
+  AssertCurrentThreadOwnsQuotaMutex();
+  MOZ_ASSERT(mGroupInfo->mPersistenceType == PERSISTENCE_TYPE_DEFAULT);
+  MOZ_ASSERT(!mPersisted);
+
+  mPersisted = true;
+
+  // Remove Usage from GroupInfo
+  AssertNoUnderflow(mGroupInfo->mUsage, mUsage);
+  mGroupInfo->mUsage -= mUsage;
+}
+
 already_AddRefed<OriginInfo>
 GroupInfo::LockedGetOriginInfo(const nsACString& aOrigin)
 {
   AssertCurrentThreadOwnsQuotaMutex();
 
   for (RefPtr<OriginInfo>& originInfo : mOriginInfos) {
     if (originInfo->mOrigin == aOrigin) {
       RefPtr<OriginInfo> result = originInfo;
@@ -5444,35 +5514,39 @@ void
 GroupInfo::LockedAddOriginInfo(OriginInfo* aOriginInfo)
 {
   AssertCurrentThreadOwnsQuotaMutex();
 
   NS_ASSERTION(!mOriginInfos.Contains(aOriginInfo),
                "Replacing an existing entry!");
   mOriginInfos.AppendElement(aOriginInfo);
 
-  AssertNoOverflow(mUsage, aOriginInfo->mUsage);
-  mUsage += aOriginInfo->mUsage;
+  if (!aOriginInfo->LockedPersisted()) {
+    AssertNoOverflow(mUsage, aOriginInfo->mUsage);
+    mUsage += aOriginInfo->mUsage;
+  }
 
   QuotaManager* quotaManager = QuotaManager::Get();
   MOZ_ASSERT(quotaManager);
 
   AssertNoOverflow(quotaManager->mTemporaryStorageUsage, aOriginInfo->mUsage);
   quotaManager->mTemporaryStorageUsage += aOriginInfo->mUsage;
 }
 
 void
 GroupInfo::LockedRemoveOriginInfo(const nsACString& aOrigin)
 {
   AssertCurrentThreadOwnsQuotaMutex();
 
   for (uint32_t index = 0; index < mOriginInfos.Length(); index++) {
     if (mOriginInfos[index]->mOrigin == aOrigin) {
-      AssertNoUnderflow(mUsage, mOriginInfos[index]->mUsage);
-      mUsage -= mOriginInfos[index]->mUsage;
+      if (!mOriginInfos[index]->LockedPersisted()) {
+        AssertNoUnderflow(mUsage, mOriginInfos[index]->mUsage);
+        mUsage -= mOriginInfos[index]->mUsage;
+      }
 
       QuotaManager* quotaManager = QuotaManager::Get();
       MOZ_ASSERT(quotaManager);
 
       AssertNoUnderflow(quotaManager->mTemporaryStorageUsage,
                         mOriginInfos[index]->mUsage);
       quotaManager->mTemporaryStorageUsage -= mOriginInfos[index]->mUsage;
 
@@ -5489,18 +5563,20 @@ GroupInfo::LockedRemoveOriginInfos()
   AssertCurrentThreadOwnsQuotaMutex();
 
   QuotaManager* quotaManager = QuotaManager::Get();
   MOZ_ASSERT(quotaManager);
 
   for (uint32_t index = mOriginInfos.Length(); index > 0; index--) {
     OriginInfo* originInfo = mOriginInfos[index - 1];
 
-    AssertNoUnderflow(mUsage, originInfo->mUsage);
-    mUsage -= originInfo->mUsage;
+    if (!originInfo->LockedPersisted()) {
+      AssertNoUnderflow(mUsage, originInfo->mUsage);
+      mUsage -= originInfo->mUsage;
+    }
 
     AssertNoUnderflow(quotaManager->mTemporaryStorageUsage, originInfo->mUsage);
     quotaManager->mTemporaryStorageUsage -= originInfo->mUsage;
 
     mOriginInfos.RemoveElementAt(index - 1);
   }
 }
 
--- a/dom/quota/QuotaManager.h
+++ b/dom/quota/QuotaManager.h
@@ -142,17 +142,18 @@ public:
     return mTemporaryStorageInitialized;
   }
 
   void
   InitQuotaForOrigin(PersistenceType aPersistenceType,
                      const nsACString& aGroup,
                      const nsACString& aOrigin,
                      uint64_t aUsageBytes,
-                     int64_t aAccessTime);
+                     int64_t aAccessTime,
+                     bool aPersisted);
 
   void
   DecreaseUsageForOrigin(PersistenceType aPersistenceType,
                          const nsACString& aGroup,
                          const nsACString& aOrigin,
                          int64_t aSize);
 
   void
@@ -455,16 +456,17 @@ private:
   nsresult
   InitializeRepository(PersistenceType aPersistenceType);
 
   nsresult
   InitializeOrigin(PersistenceType aPersistenceType,
                    const nsACString& aGroup,
                    const nsACString& aOrigin,
                    int64_t aAccessTime,
+                   bool aPersisted,
                    nsIFile* aDirectory);
 
   void
   CheckTemporaryStorageLimits();
 
   void
   DeleteFilesForOrigin(PersistenceType aPersistenceType,
                        const nsACString& aOrigin);
--- a/dom/webbrowserpersist/PWebBrowserPersistDocument.ipdl
+++ b/dom/webbrowserpersist/PWebBrowserPersistDocument.ipdl
@@ -1,18 +1,20 @@
 /* -*- Mode: IDL; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 /* 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 protocol PContent;
 include protocol PWebBrowserPersistResources;
 include protocol PWebBrowserPersistSerialize;
+include protocol PFileDescriptorSet;
+include protocol PSendStream;
 
-include InputStreamParams;
+include IPCStream;
 
 namespace mozilla {
 
 // nsIWebBrowserPersistDocument has attributes which can be read
 // synchronously.  To avoid using sync IPC for them, the actor sends
 // this structure from the child to the parent before the parent actor
 // is exposed to XPCOM.
 struct WebBrowserPersistDocumentAttrs {
@@ -55,18 +57,17 @@ protocol PWebBrowserPersistDocument {
   manages PWebBrowserPersistSerialize;
 
 parent:
   // The actor isn't exposed to XPCOM until after it gets one of these
   // two messages; see also the state transition rules.  The message
   // is either a response to the constructor (if it was parent->child)
   // or sent after it (if it was child->parent).
   async Attributes(WebBrowserPersistDocumentAttrs aAttrs,
-                   OptionalInputStreamParams postData,
-                   FileDescriptor[] postFiles);
+                   OptionalIPCStream stream);
   async InitFailure(nsresult aStatus);
 
 child:
   async SetPersistFlags(uint32_t aNewFlags);
   async PWebBrowserPersistResources();
   async PWebBrowserPersistSerialize(WebBrowserPersistURIMap aMap,
                                     nsCString aRequestedContentType,
                                     uint32_t aEncoderFlags,
--- a/dom/webbrowserpersist/WebBrowserPersistDocumentChild.cpp
+++ b/dom/webbrowserpersist/WebBrowserPersistDocumentChild.cpp
@@ -1,17 +1,18 @@
 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  *
  * 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 "WebBrowserPersistDocumentChild.h"
 
-#include "mozilla/ipc/InputStreamUtils.h"
+#include "mozilla/dom/ContentChild.h"
+#include "mozilla/ipc/IPCStreamUtils.h"
 #include "nsIDocument.h"
 #include "nsIInputStream.h"
 #include "WebBrowserPersistLocalDocument.h"
 #include "WebBrowserPersistResourcesChild.h"
 #include "WebBrowserPersistSerializeChild.h"
 
 namespace mozilla {
 
@@ -37,18 +38,16 @@ WebBrowserPersistDocumentChild::Start(ns
     MOZ_ASSERT(!mDocument);
     if (!aDocument) {
         SendInitFailure(NS_ERROR_FAILURE);
         return;
     }
 
     WebBrowserPersistDocumentAttrs attrs;
     nsCOMPtr<nsIInputStream> postDataStream;
-    OptionalInputStreamParams postData;
-    nsTArray<FileDescriptor> postFiles;
 #define ENSURE(e) do {           \
         nsresult rv = (e);       \
         if (NS_FAILED(rv)) {     \
             SendInitFailure(rv); \
             return;              \
         }                        \
     } while(0)
     ENSURE(aDocument->GetIsPrivate(&(attrs.isPrivate())));
@@ -57,22 +56,24 @@ WebBrowserPersistDocumentChild::Start(ns
     ENSURE(aDocument->GetContentType(attrs.contentType()));
     ENSURE(aDocument->GetCharacterSet(attrs.characterSet()));
     ENSURE(aDocument->GetTitle(attrs.title()));
     ENSURE(aDocument->GetReferrer(attrs.referrer()));
     ENSURE(aDocument->GetContentDisposition(attrs.contentDisposition()));
     ENSURE(aDocument->GetCacheKey(&(attrs.cacheKey())));
     ENSURE(aDocument->GetPersistFlags(&(attrs.persistFlags())));
     ENSURE(aDocument->GetPostData(getter_AddRefs(postDataStream)));
-    ipc::SerializeInputStream(postDataStream,
-                              postData,
-                              postFiles);
 #undef ENSURE
+
+    mozilla::ipc::AutoIPCStream autoStream;
+    autoStream.Serialize(postDataStream,
+                         static_cast<mozilla::dom::ContentChild*>(Manager()));
+
     mDocument = aDocument;
-    SendAttributes(attrs, postData, postFiles);
+    SendAttributes(attrs, autoStream.TakeOptionalValue());
 }
 
 mozilla::ipc::IPCResult
 WebBrowserPersistDocumentChild::RecvSetPersistFlags(const uint32_t& aNewFlags)
 {
     mDocument->SetPersistFlags(aNewFlags);
     return IPC_OK();
 }
--- a/dom/webbrowserpersist/WebBrowserPersistDocumentParent.cpp
+++ b/dom/webbrowserpersist/WebBrowserPersistDocumentParent.cpp
@@ -1,17 +1,17 @@
 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  *
  * 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 "WebBrowserPersistDocumentParent.h"
 
-#include "mozilla/ipc/InputStreamUtils.h"
+#include "mozilla/ipc/IPCStreamUtils.h"
 #include "mozilla/dom/PContentParent.h"
 #include "nsIInputStream.h"
 #include "nsThreadUtils.h"
 #include "WebBrowserPersistResourcesParent.h"
 #include "WebBrowserPersistSerializeParent.h"
 #include "WebBrowserPersistRemoteDocument.h"
 
 namespace mozilla {
@@ -55,22 +55,20 @@ WebBrowserPersistDocumentParent::ActorDe
 WebBrowserPersistDocumentParent::~WebBrowserPersistDocumentParent()
 {
     MOZ_RELEASE_ASSERT(!mReflection);
     MOZ_ASSERT(!mOnReady);
 }
 
 mozilla::ipc::IPCResult
 WebBrowserPersistDocumentParent::RecvAttributes(const Attrs& aAttrs,
-                                                const OptionalInputStreamParams& aPostData,
-                                                nsTArray<FileDescriptor>&& aPostFiles)
+                                                const OptionalIPCStream& aPostStream)
 {
     // Deserialize the postData unconditionally so that fds aren't leaked.
-    nsCOMPtr<nsIInputStream> postData =
-        ipc::DeserializeInputStream(aPostData, aPostFiles);
+    nsCOMPtr<nsIInputStream> postData = mozilla::ipc::DeserializeIPCStream(aPostStream);
     if (!mOnReady || mReflection) {
         return IPC_FAIL_NO_REASON(this);
     }
     mReflection = new WebBrowserPersistRemoteDocument(this, aAttrs, postData);
     RefPtr<WebBrowserPersistRemoteDocument> reflection = mReflection;
     mOnReady->OnDocumentReady(reflection);
     mOnReady = nullptr;
     return IPC_OK();
--- a/dom/webbrowserpersist/WebBrowserPersistDocumentParent.h
+++ b/dom/webbrowserpersist/WebBrowserPersistDocumentParent.h
@@ -42,18 +42,17 @@ public:
     // is still in the START state (or is unconstructed).
     void SetOnReady(nsIWebBrowserPersistDocumentReceiver* aOnReady);
 
     using Attrs = WebBrowserPersistDocumentAttrs;
 
     // IPDL methods:
     virtual mozilla::ipc::IPCResult
     RecvAttributes(const Attrs& aAttrs,
-                   const OptionalInputStreamParams& aPostData,
-                   nsTArray<FileDescriptor>&& aPostFiles) override;
+                   const OptionalIPCStream& aPostStream) override;
     virtual mozilla::ipc::IPCResult
     RecvInitFailure(const nsresult& aFailure) override;
 
     virtual PWebBrowserPersistResourcesParent*
     AllocPWebBrowserPersistResourcesParent() override;
     virtual bool
     DeallocPWebBrowserPersistResourcesParent(PWebBrowserPersistResourcesParent* aActor) override;
 
--- a/gfx/2d/DrawEventRecorder.h
+++ b/gfx/2d/DrawEventRecorder.h
@@ -129,16 +129,24 @@ public:
 
   /**
    * Wipes the internal recording buffer, but the recorder does NOT forget which
    * objects it has recorded. This can be used so that a recording can be copied
    * and processed in chunks, releasing memory as it goes.
    */
   void WipeRecording();
 
+  /**
+   * Gets a readable reference of the underlying stream, reset to the beginning.
+   */
+  std::istream& GetInputStream() {
+    mMemoryStream.seekg(0);
+    return mMemoryStream;
+  }
+
 private:
   ~DrawEventRecorderMemory() {};
 
   void Flush() final;
 
   std::stringstream mMemoryStream;
 };
 
--- a/gfx/gl/SharedSurfaceD3D11Interop.cpp
+++ b/gfx/gl/SharedSurfaceD3D11Interop.cpp
@@ -1,16 +1,17 @@
 /* -*- Mode: c++; c-basic-offset: 4; indent-tabs-mode: nil; tab-width: 40; -*- */
 /* 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 "SharedSurfaceD3D11Interop.h"
 
 #include <d3d11.h>
+#include <d3d11_1.h>
 #include "gfxPrefs.h"
 #include "GLContext.h"
 #include "WGLLibrary.h"
 #include "nsPrintfCString.h"
 #include "mozilla/gfx/DeviceManagerDx.h"
 
 namespace mozilla {
 namespace gl {
@@ -109,54 +110,111 @@ while (!done) {
       <direct3d renders to the render targets and presents
        the results on the screen>
 }
 */
 
 ////////////////////////////////////////////////////////////////////////////////
 // DXInterop2Device
 
+class ScopedContextState final
+{
+    ID3D11DeviceContext1* const mD3DContext;
+    RefPtr<ID3DDeviceContextState> mOldContextState;
+
+public:
+    ScopedContextState(ID3D11DeviceContext1* d3dContext,
+                       ID3DDeviceContextState* newContextState)
+        : mD3DContext(d3dContext)
+        , mOldContextState(nullptr)
+    {
+        if (!mD3DContext)
+            return;
+
+        mD3DContext->SwapDeviceContextState(newContextState,
+                                            getter_AddRefs(mOldContextState));
+    }
+
+    ~ScopedContextState()
+    {
+        if (!mD3DContext)
+            return;
+
+        mD3DContext->SwapDeviceContextState(mOldContextState, nullptr);
+    }
+};
+
 class DXInterop2Device : public RefCounted<DXInterop2Device>
 {
 public:
     MOZ_DECLARE_REFCOUNTED_TYPENAME(DXInterop2Device)
 
     WGLLibrary* const mWGL;
     const RefPtr<ID3D11Device> mD3D; // Only needed for lifetime guarantee.
     const HANDLE mInteropDevice;
     GLContext* const mGL;
 
+    // AMD workaround.
+    const RefPtr<ID3D11DeviceContext1> mD3DContext;
+    const RefPtr<ID3DDeviceContextState> mContextState;
+
     static already_AddRefed<DXInterop2Device> Open(WGLLibrary* wgl, GLContext* gl)
     {
         MOZ_ASSERT(wgl->HasDXInterop2());
 
         const RefPtr<ID3D11Device> d3d = gfx::DeviceManagerDx::Get()->GetContentDevice();
         if (!d3d) {
             gfxCriticalNote << "DXInterop2Device::Open: Failed to create D3D11 device.";
             return nullptr;
         }
 
         if (!gl->MakeCurrent())
             return nullptr;
 
+        RefPtr<ID3D11DeviceContext1> d3dContext;
+        RefPtr<ID3DDeviceContextState> contextState;
+        if (gl->WorkAroundDriverBugs() && gl->Vendor() == GLVendor::ATI) {
+            // AMD calls ID3D10Device::Flush, so we need to be in ID3D10Device mode.
+            RefPtr<ID3D11Device1> d3d11_1;
+            auto hr = d3d->QueryInterface(__uuidof(ID3D11Device1),
+                                          getter_AddRefs(d3d11_1));
+            if (!SUCCEEDED(hr))
+                return nullptr;
+
+            d3d11_1->GetImmediateContext1(getter_AddRefs(d3dContext));
+            MOZ_ASSERT(d3dContext);
+
+            const D3D_FEATURE_LEVEL featureLevel10_0 = D3D_FEATURE_LEVEL_10_0;
+            hr = d3d11_1->CreateDeviceContextState(0, &featureLevel10_0, 1,
+                                                   D3D11_SDK_VERSION,
+                                                   __uuidof(ID3D10Device), nullptr,
+                                                   getter_AddRefs(contextState));
+            if (!SUCCEEDED(hr))
+                return nullptr;
+        }
+
         const auto interopDevice = wgl->mSymbols.fDXOpenDeviceNV(d3d);
         if (!interopDevice) {
             gfxCriticalNote << "DXInterop2Device::Open: DXOpenDevice failed.";
             return nullptr;
         }
 
-        return MakeAndAddRef<DXInterop2Device>(wgl, d3d, interopDevice, gl);
+        return MakeAndAddRef<DXInterop2Device>(wgl, d3d, interopDevice, gl, d3dContext,
+                                               contextState);
     }
 
     DXInterop2Device(WGLLibrary* wgl, ID3D11Device* d3d, HANDLE interopDevice,
-                     GLContext* gl)
+                     GLContext* gl, ID3D11DeviceContext1* d3dContext,
+                     ID3DDeviceContextState* contextState)
         : mWGL(wgl)
         , mD3D(d3d)
         , mInteropDevice(interopDevice)
         , mGL(gl)
+        , mD3DContext(d3dContext)
+        , mContextState(contextState)
     { }
 
     ~DXInterop2Device() {
         const auto isCurrent = mGL->MakeCurrent();
 
         if (mWGL->mSymbols.fDXCloseDeviceNV(mInteropDevice))
             return;
 
@@ -171,33 +229,35 @@ public:
     }
 
     HANDLE RegisterObject(void* d3dObject, GLuint name, GLenum type,
                           GLenum access) const
     {
         if (!mGL->MakeCurrent())
             return nullptr;
 
-        const auto& ret = mWGL->mSymbols.fDXRegisterObjectNV(mInteropDevice, d3dObject,
-                                                             name, type, access);
+        const ScopedContextState autoCS(mD3DContext, mContextState);
+        const auto ret = mWGL->mSymbols.fDXRegisterObjectNV(mInteropDevice, d3dObject,
+                                                            name, type, access);
         if (ret)
             return ret;
 
         const uint32_t error = GetLastError();
         const nsPrintfCString errorMessage("wglDXRegisterObject(0x%p, 0x%p, %u, 0x%04x,"
                                            " 0x%04x) failed: GetLastError(): %u\n",
                                            mInteropDevice, d3dObject, name, type, access,
                                            error);
         gfxCriticalNote << errorMessage.BeginReading();
         return nullptr;
     }
 
     bool UnregisterObject(HANDLE lockHandle) const {
         const auto isCurrent = mGL->MakeCurrent();
 
+        const ScopedContextState autoCS(mD3DContext, mContextState);
         if (mWGL->mSymbols.fDXUnregisterObjectNV(mInteropDevice, lockHandle))
             return true;
 
         if (!isCurrent) {
             // That shouldn't have failed.
             const uint32_t error = GetLastError();
             const nsPrintfCString errorMessage("wglDXUnregisterObject(0x%p, 0x%p) failed:"
                                                " GetLastError(): %u\n",
--- a/gfx/layers/basic/BasicLayersImpl.cpp
+++ b/gfx/layers/basic/BasicLayersImpl.cpp
@@ -160,23 +160,17 @@ FillRectWithMask(DrawTarget* aDT,
     if (aSurfaceTransform) {
       transform = (*aSurfaceTransform) * transform;
     }
 
     InlineTranslator* translator = new InlineTranslator(aDT, transform);
     SourceSurfaceRecording* ss = static_cast<SourceSurfaceRecording*>(aSurface);
     DrawEventRecorderMemory* mr = static_cast<DrawEventRecorderMemory*>(ss->mRecorder.get());
 
-    size_t size = mr->RecordingSize();
-    char* buffer = new char[size];
-    mr->CopyRecording(buffer, size);
-    std::istringstream recording(std::string(buffer, size));
-
-    delete [] buffer;
-    translator->TranslateRecording(recording);
+    translator->TranslateRecording(mr->GetInputStream());
 
     aDT->SetTransform(oldTransform);
     aDT->PopClip();
     return;
   }
 
   aDT->FillRect(aRect,
                 SurfacePattern(aSurface, aExtendMode,
--- a/gfx/thebes/DeviceManagerDx.cpp
+++ b/gfx/thebes/DeviceManagerDx.cpp
@@ -362,32 +362,58 @@ DeviceManagerDx::CreateCompositorDevice(
       false,
       textureSharingWorks,
       featureLevel,
       DxgiAdapterDesc::From(desc)));
   }
   mCompositorDevice->SetExceptionMode(0);
 }
 
+//#define BREAK_ON_D3D_ERROR
+
 bool
 DeviceManagerDx::CreateDevice(IDXGIAdapter* aAdapter,
                                  D3D_DRIVER_TYPE aDriverType,
                                  UINT aFlags,
                                  HRESULT& aResOut,
                                  RefPtr<ID3D11Device>& aOutDevice)
 {
+#ifdef BREAK_ON_D3D_ERROR
+  aFlags |= D3D11_CREATE_DEVICE_DEBUG;
+#endif
+
   MOZ_SEH_TRY {
     aResOut = sD3D11CreateDeviceFn(
       aAdapter, aDriverType, nullptr,
       aFlags,
       mFeatureLevels.Elements(), mFeatureLevels.Length(),
       D3D11_SDK_VERSION, getter_AddRefs(aOutDevice), nullptr, nullptr);
   } MOZ_SEH_EXCEPT (EXCEPTION_EXECUTE_HANDLER) {
     return false;
   }
+
+#ifdef BREAK_ON_D3D_ERROR
+  do {
+    if (!aOutDevice)
+      break;
+
+    RefPtr<ID3D11Debug> debug;
+    if(!SUCCEEDED( aOutDevice->QueryInterface(__uuidof(ID3D11Debug), getter_AddRefs(debug)) ))
+      break;
+
+    RefPtr<ID3D11InfoQueue> infoQueue;
+    if(!SUCCEEDED( debug->QueryInterface(__uuidof(ID3D11InfoQueue), getter_AddRefs(infoQueue)) ))
+      break;
+
+    infoQueue->SetBreakOnSeverity(D3D11_MESSAGE_SEVERITY_CORRUPTION, true);
+    infoQueue->SetBreakOnSeverity(D3D11_MESSAGE_SEVERITY_ERROR, true);
+    infoQueue->SetBreakOnSeverity(D3D11_MESSAGE_SEVERITY_WARNING, true);
+  } while (false);
+#endif
+
   return true;
 }
 
 void
 DeviceManagerDx::CreateWARPCompositorDevice()
 {
   ScopedGfxFeatureReporter reporterWARP("D3D11-WARP", gfxPrefs::LayersD3D11ForceWARP());
   FeatureState& d3d11 = gfxConfig::GetFeature(Feature::D3D11_COMPOSITING);
--- a/gfx/thebes/gfxSkipChars.cpp
+++ b/gfx/thebes/gfxSkipChars.cpp
@@ -14,18 +14,22 @@ struct SkippedRangeStartComparator
         return (mOffset < aRange.Start()) ? -1 : 1;
     }
 };
 
 void
 gfxSkipCharsIterator::SetOriginalOffset(int32_t aOffset)
 {
     aOffset += mOriginalStringToSkipCharsOffset;
-    NS_ASSERTION(uint32_t(aOffset) <= mSkipChars->mCharCount,
-                 "Invalid offset");
+    if (MOZ_UNLIKELY(uint32_t(aOffset) > mSkipChars->mCharCount)) {
+        gfxCriticalError() <<
+            "invalid offset " << aOffset <<
+            " for gfxSkipChars length " << mSkipChars->mCharCount;
+        aOffset = mSkipChars->mCharCount;
+    }
 
     mOriginalStringOffset = aOffset;
 
     const uint32_t rangeCount = mSkipChars->mRanges.Length();
     if (rangeCount == 0) {
         mSkippedStringOffset = aOffset;
         return;
     }
--- a/ipc/glue/IPCStreamUtils.cpp
+++ b/ipc/glue/IPCStreamUtils.cpp
@@ -115,16 +115,31 @@ SerializeInputStreamWithFdsParent(nsIInp
     if (fdSet) {
       streamWithFds.optionalFds() = fdSet;
     }
   }
 }
 
 template<typename M>
 void
+SerializeInputStreamWithFdsParent(nsIInputStream* aStream,
+                                  OptionalIPCStream& aValue,
+                                  M* aManager)
+{
+  if (!aStream) {
+    aValue = void_t();
+    return;
+  }
+
+  aValue = IPCStream();
+  SerializeInputStreamWithFdsParent(aStream, aValue.get_IPCStream(), aManager);
+}
+
+template<typename M>
+void
 SerializeInputStream(nsIInputStream* aStream, IPCStream& aValue, M* aManager)
 {
   MOZ_ASSERT(aStream);
   MOZ_ASSERT(aManager);
 
   // If a stream is known to be larger than 1MB, prefer sending it in chunks.
   const uint64_t kTooLargeStream = 1024 * 1024;
 
@@ -173,18 +188,17 @@ SerializeInputStream(nsIInputStream* aSt
                      M* aManager)
 {
   if (!aStream) {
     aValue = void_t();
     return;
   }
 
   aValue = IPCStream();
-  SerializeInputStream(aStream, aValue.get_IPCStream(),
-                             aManager);
+  SerializeInputStream(aStream, aValue.get_IPCStream(), aManager);
 }
 
 void
 CleanupIPCStream(IPCStream& aValue, bool aConsumedByIPC)
 {
   if (aValue.type() == IPCStream::T__None) {
     return;
   }
@@ -373,17 +387,17 @@ AutoIPCStream::~AutoIPCStream()
   } else {
     CleanupIPCStream(*mOptionalValue, mTaken);
   }
 }
 
 void
 AutoIPCStream::Serialize(nsIInputStream* aStream, dom::nsIContentChild* aManager)
 {
-  MOZ_ASSERT(aStream);
+  MOZ_ASSERT(aStream || !mValue);
   MOZ_ASSERT(aManager);
   MOZ_ASSERT(mValue || mOptionalValue);
   MOZ_ASSERT(!mTaken);
   MOZ_ASSERT(!IsSet());
 
   if (mValue) {
     SerializeInputStream(aStream, *mValue, aManager);
     AssertValidValueToTake(*mValue);
@@ -391,17 +405,17 @@ AutoIPCStream::Serialize(nsIInputStream*
     SerializeInputStream(aStream, *mOptionalValue, aManager);
     AssertValidValueToTake(*mOptionalValue);
   }
 }
 
 void
 AutoIPCStream::Serialize(nsIInputStream* aStream, PBackgroundChild* aManager)
 {
-  MOZ_ASSERT(aStream);
+  MOZ_ASSERT(aStream || !mValue);
   MOZ_ASSERT(aManager);
   MOZ_ASSERT(mValue || mOptionalValue);
   MOZ_ASSERT(!mTaken);
   MOZ_ASSERT(!IsSet());
 
   if (mValue) {
     SerializeInputStream(aStream, *mValue, aManager);
     AssertValidValueToTake(*mValue);
@@ -409,17 +423,17 @@ AutoIPCStream::Serialize(nsIInputStream*
     SerializeInputStream(aStream, *mOptionalValue, aManager);
     AssertValidValueToTake(*mOptionalValue);
   }
 }
 
 void
 AutoIPCStream::Serialize(nsIInputStream* aStream, dom::PContentParent* aManager)
 {
-  MOZ_ASSERT(aStream);
+  MOZ_ASSERT(aStream || !mValue);
   MOZ_ASSERT(aManager);
   MOZ_ASSERT(mValue || mOptionalValue);
   MOZ_ASSERT(!mTaken);
   MOZ_ASSERT(!IsSet());
 
   if (mValue) {
     SerializeInputStreamWithFdsParent(aStream, *mValue, aManager);
     AssertValidValueToTake(*mValue);
@@ -427,17 +441,17 @@ AutoIPCStream::Serialize(nsIInputStream*
     SerializeInputStreamWithFdsParent(aStream, *mOptionalValue, aManager);
     AssertValidValueToTake(*mOptionalValue);
   }
 }
 
 void
 AutoIPCStream::Serialize(nsIInputStream* aStream, PBackgroundParent* aManager)
 {
-  MOZ_ASSERT(aStream);
+  MOZ_ASSERT(aStream || !mValue);
   MOZ_ASSERT(aManager);
   MOZ_ASSERT(mValue || mOptionalValue);
   MOZ_ASSERT(!mTaken);
   MOZ_ASSERT(!IsSet());
 
   if (mValue) {
     SerializeInputStreamWithFdsParent(aStream, *mValue, aManager);
     AssertValidValueToTake(*mValue);
--- a/js/src/jit/FlowAliasAnalysis.cpp
+++ b/js/src/jit/FlowAliasAnalysis.cpp
@@ -458,23 +458,30 @@ FlowAliasAnalysis::analyze()
 
         DumpBlockStart(*block);
 
         if (!computeBlockStores(*block))
             return false;
         if (!stores_->maybeFreePredecessorBlocks(*block))
             return false;
 
-        if (block->isLoopHeader())
-            loop_ = new(alloc()) LoopInfo(alloc(), loop_, *block);
-
         for (MPhiIterator def(block->phisBegin()), end(block->phisEnd()); def != end; ++def)
             def->setId(newId++);
 
         BlockStoreInfo& blockInfo = stores_->current();
+
+        // When the store dependencies is empty it means we have a disconnected
+        // graph. Those blocks will never get reached but it is only fixed up
+        // after GVN. Don't run AA on those blocks.
+        if (blockInfo.length() == 0)
+            continue;
+
+        if (block->isLoopHeader())
+            loop_ = new(alloc()) LoopInfo(alloc(), loop_, *block);
+
         for (MInstructionIterator def(block->begin()), end(block->begin(block->lastIns()));
                 def != end;
                 ++def)
         {
             def->setId(newId++);
 
             // For the purposes of alias analysis, all recoverable operations
             // are treated as effect free as the memory represented by these
@@ -583,16 +590,24 @@ FlowAliasAnalysis::processDeferredLoads(
         if (!isLoopInvariant(load, store, &loopinvariant))
             return false;
         MDefinitionVector &loopInvariantDependency =
             stores_->get(store->block()->loopPredecessor());
 
         DumpLoopInvariant(load, info->loopHeader(), /* loopinvariant = */ loopinvariant,
                           loopInvariantDependency);
 
+        // When the store dependencies is empty it means we have a disconnected
+        // graph. Those blocks will never get reached but it is only fixed up
+        // after GVN. Don't improve dependency for those loads.
+        if (loopInvariantDependency.length() == 0) {
+            load->setDependency(store);
+            continue;
+        }
+
         if (loopinvariant) {
             if (!improveDependency(load, loopInvariantDependency, output_))
                 return false;
             saveLoadDependency(load, output_);
 
             // If possible defer when better loop information is present.
             if (deferImproveDependency(output_)) {
                 if (!loop_->loopinvariant().append(load))
--- a/js/src/tests/js1_5/extensions/regress-306738.js
+++ b/js/src/tests/js1_5/extensions/regress-306738.js
@@ -15,11 +15,14 @@ printStatus (summary);
 actual = uneval(
   {
     get foo()
     {
       return "foo";
     }
   });
 
-expect = '({get foo() {return "foo";}})';
+expect = '({get foo()\n\
+    {\n\
+      return "foo";\n\
+    }})';
  
 compareSource(expect, actual, summary);
--- a/js/src/tests/js1_5/extensions/regress-313803.js
+++ b/js/src/tests/js1_5/extensions/regress-313803.js
@@ -14,11 +14,14 @@ printStatus (summary);
 
 var func = function ff() {
   obj = { get foo() { return "foo"; }};
   return 1;
 };
 
 actual = uneval(func);
 
-expect = '(function ff() {obj = {get foo () {return "foo";}};return 1;})';
+expect = '(function ff() {\n\
+  obj = { get foo() { return "foo"; }};\n\
+  return 1;\n\
+})';
 
 compareSource(expect, actual, summary);
--- a/js/src/tests/js1_5/extensions/regress-333541.js
+++ b/js/src/tests/js1_5/extensions/regress-333541.js
@@ -13,45 +13,45 @@ printBugNumber(BUGNUMBER);
 printStatus (summary);
  
 function a(){
   return 1..toSource();
 }
 
 try
 {
-  expect = 'function a() {\n    return 1..toSource();\n}';
+  expect = 'function a(){\n  return 1..toSource();\n}';
   actual = a.toString();
   compareSource(expect, actual, summary + ': 1');
 }
 catch(ex)
 {
   actual = ex + '';
   reportCompare(expect, actual, summary + ': 1');
 }
 
 try
 {
-  expect = 'function a() {return 1..toSource();}';
+  expect = 'function a(){\n  return 1..toSource();\n}';
   actual = a.toSource();
   compareSource(expect, actual, summary + ': 2');
 }
 catch(ex)
 {
   actual = ex + '';
   reportCompare(expect, actual, summary + ': 2');
 }
 
 expect = a;
 actual = a.valueOf();
 reportCompare(expect, actual, summary + ': 3');
 
 try
 {
-  expect = 'function a() {\n    return 1..toSource();\n}';
+  expect = 'function a(){\n  return 1..toSource();\n}';
   actual = "" + a;
   compareSource(expect, actual, summary + ': 4');
 }
 catch(ex)
 {
   actual = ex + '';
   reportCompare(expect, actual, summary + ': 4');
 }
--- a/js/src/tests/js1_5/extensions/regress-374589.js
+++ b/js/src/tests/js1_5/extensions/regress-374589.js
@@ -19,16 +19,16 @@ function test()
 {
   enterFunc ('test');
   printBugNumber(BUGNUMBER);
   printStatus (summary);
  
   var f = function () {
     try { } catch(x if true) { } catch(y) { } finally { this.a.b; } };
 
-  expect = 'function () { try { } catch(x if true) { } catch(y) { } ' +
-    'finally { this.a.b; } }';
+  expect = 'function () {\n\
+    try { } catch(x if true) { } catch(y) { } finally { this.a.b; } }';
 
   actual = f + '';
   compareSource(expect, actual, summary);
 
   exitFunc ('test');
 }
--- a/js/src/tests/js1_5/extensions/regress-375801.js
+++ b/js/src/tests/js1_5/extensions/regress-375801.js
@@ -15,17 +15,17 @@ test();
 //-----------------------------------------------------------------------------
 
 function test()
 {
   enterFunc ('test');
   printBugNumber(BUGNUMBER);
   printStatus (summary);
  
-  expect = '({a: (void 0)})'
+  expect = '({a:(void 0)})'
   actual = uneval({a: undefined})
   compareSource(expect, actual, summary + ': uneval');
 
   expect = 'function() {({a: undefined});}';
   actual = (function() {({a: undefined});}).toString();
   compareSource(expect, actual, summary + ': toString');
 
   expect = '(function () {({a: undefined});})';
--- a/js/src/tests/js1_5/extensions/regress-381211.js
+++ b/js/src/tests/js1_5/extensions/regress-381211.js
@@ -16,14 +16,14 @@ test();
 //-----------------------------------------------------------------------------
 
 function test()
 {
   enterFunc ('test');
   printBugNumber(BUGNUMBER);
   printStatus (summary);
  
-  expect = '( { get x() {} } )';
+  expect = '({get x(){}})';
   actual = uneval({get x(){}});
   compareSource(expect, actual, summary);
 
   exitFunc ('test');
 }
--- a/js/src/tests/js1_5/extensions/regress-381304.js
+++ b/js/src/tests/js1_5/extensions/regress-381304.js
@@ -24,46 +24,46 @@ function test()
 
   print('1');
 
   obj = {
     set inn(value) {this.for = value;}, 
     get inn() {return this.for;}
   };
 
-  expect = '({get inn() { return this.for; }, set inn(value) { this.for = value; } })';
+  expect = '({get inn() {return this.for;}, set inn(value) {this.for = value;}})';
   actual = obj.toSource();
   compareSource(expect, actual, summary + ': 1');
 
   print('2');
 
   obj = {
     set in(value) {this.for = value;}, 
     get in() {return this.for;}
   };
 
-  expect = '({ get in() { return this.for; }, set in(value) { this.for = value; } })';
+  expect = '({get in() {return this.for;}, set in(value) {this.for = value;}})';
   actual = obj.toSource();
   compareSource(expect, actual, summary + ': 2');
 
   print('3');
 
   obj = {
     set inn(value) {this.for = value;}, 
     get in() {return this.for;}
   };
 
-  expect = '({ set inn(value) { this.for = value; }, get in() { return this.for; } })';
+  expect = '({set inn(value) {this.for = value;}, get in() {return this.for;}})';
   actual = obj.toSource();
   compareSource(expect, actual, summary + ': 4');
 
   print('4');
 
   obj = {
     set in(value) {this.for = value;}, 
     get inn() {return this.for;}
   };
 
-  expect = ' ({ set in(value) { this.for = value; }, get inn() { return this.for; } })';
+  expect = '({set in(value) {this.for = value;}, get inn() {return this.for;}})';
   actual = obj.toSource();
   compareSource(expect, actual, summary + ': 5');
   exitFunc ('test');
 }
--- a/js/src/tests/js1_7/extensions/regress-367629.js
+++ b/js/src/tests/js1_7/extensions/regress-367629.js
@@ -27,12 +27,20 @@ function test()
       get: encodeURI,
       enumerable: true,
       configurable: true
     });
 
   expect = '({get h() {[native code]}})';
   actual = uneval(a);      
 
+  // Native function syntax:
+  // `function IdentifierName_opt ( FormalParameters ) { [ native code ] }`
+
+  // The placement of whitespace characters in the native function's body is
+  // implementation-dependent, so we need to replace those for this test.
+  var re = new RegExp(["\\{", "\\[", "native", "code", "\\]", "\\}"].join("\\s*"));
+  actual = actual.replace(re, "{[native code]}");
+
   compareSource(expect, actual, summary);
 
   exitFunc ('test');
 }
--- a/js/src/tests/js1_7/extensions/regress-379566.js
+++ b/js/src/tests/js1_7/extensions/regress-379566.js
@@ -16,19 +16,19 @@ test();
 //-----------------------------------------------------------------------------
 
 function test()
 {
   enterFunc ('test');
   printBugNumber(BUGNUMBER);
   printStatus (summary);
  
-  expect = '({ ' +
-    'get in () { return this.for; }, ' + 
-    'set in (value) { this.for = value; } ' + 
+  expect = '({' +
+    'get in() { return this.for; }, ' +
+    'set in(value) { this.for = value; }' +
     '})';
   try
   {
     var obj = eval('({ ' +
                    'get in() { return this.for; }, ' + 
                    'set in(value) { this.for = value; } ' + 
                    '})');
     actual = obj.toSource();
--- a/js/src/tests/js1_7/extensions/regress-381301.js
+++ b/js/src/tests/js1_7/extensions/regress-381301.js
@@ -18,15 +18,23 @@ test();
 function test()
 {
   enterFunc ('test');
   printBugNumber(BUGNUMBER);
   printStatus (summary);
 
   var o =
     Object.defineProperty({}, "x", { get: decodeURI, enumerable: true, configurable: true });
-  expect = '( { get x ( ) { [ native code ] } } )';
+  expect = '({get x() {[native code]}})';
   actual =  uneval(o);
 
+  // Native function syntax:
+  // `function IdentifierName_opt ( FormalParameters ) { [ native code ] }`
+
+  // The placement of whitespace characters in the native function's body is
+  // implementation-dependent, so we need to replace those for this test.
+  var re = new RegExp(["\\{", "\\[", "native", "code", "\\]", "\\}"].join("\\s*"));
+  actual = actual.replace(re, "{[native code]}");
+
   compareSource(expect, actual, summary);
 
   exitFunc ('test');
 }
--- a/js/src/tests/js1_7/extensions/regress-381303.js
+++ b/js/src/tests/js1_7/extensions/regress-381303.js
@@ -17,17 +17,17 @@ test();
 
 function test()
 {
   enterFunc ('test');
   printBugNumber(BUGNUMBER);
   printStatus (summary);
  
   var obj = {set inn(value) {this.for = value;}, get inn() {return this.for;}};
-  expect = '( { ' + 
+  expect = '({' +
     'get inn() {return this.for;}' + 
     ', ' + 
     'set inn(value) {this.for = value;}' + 
     '})';
   actual = obj.toSource();
 
   compareSource(expect, actual, summary);
 
--- a/js/src/tests/js1_8/extensions/regress-385729.js
+++ b/js/src/tests/js1_8/extensions/regress-385729.js
@@ -4,16 +4,23 @@
  * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 
 //-----------------------------------------------------------------------------
 var BUGNUMBER = 385729;
 var summary = 'uneval(eval(expression closure))';
 var actual = 'No Crash';
 var expect = 'No Crash';
 
+function normalizeSource(source) {
+  source = String(source);
+  source = source.replace(/([(){},.:\[\]])/mg, ' $1 ');
+  source = source.replace(/\s+/mg, ' ');
+
+  return source;
+}
 
 //-----------------------------------------------------------------------------
 test();
 //-----------------------------------------------------------------------------
 
 function test()
 {
   enterFunc ('test');
@@ -29,27 +36,27 @@ function test()
       actual = uneval(eval(expect));
     }
     catch(ex)
     {
       // mozilla 1.8
       expect = 'SyntaxError: missing { before function body';
       actual = ex + '';
     }
-    compareSource(expect, actual, summary);
+    compareSource(normalizeSource(expect), normalizeSource(actual), summary);
 
     expect = '({get f () /x/g})';
     try
     {
       // mozilla 1.9
       actual = uneval(eval("({get f () /x/g})"));
     }
     catch(ex)
     {
       // mozilla 1.8
       expect = 'SyntaxError: missing { before function body';
       actual = ex + '';
     }
-    compareSource(expect, actual, summary);
+    compareSource(normalizeSource(expect), normalizeSource(actual), summary);
   }
 
   exitFunc ('test');
 }
--- a/js/src/tests/shell.js
+++ b/js/src/tests/shell.js
@@ -90,38 +90,16 @@
   function StringCharCodeAt(str, index) {
     return ReflectApply(StringPrototypeCharCodeAt, str, [index]);
   }
 
   function StringEndsWith(str, needle) {
     return ReflectApply(StringPrototypeEndsWith, str, [needle]);
   }
 
-  function StringReplace(str, regexp, replacement) {
-    assertEq(typeof str === "string" && typeof regexp === "object" &&
-             typeof replacement === "function", true,
-             "StringReplace must be called with a string, a RegExp object and a function");
-
-    regexp.lastIndex = 0;
-
-    var result = "";
-    var last = 0;
-    while (true) {
-      var match = ReflectApply(RegExpPrototypeExec, regexp, [str]);
-      if (!match) {
-        result += ReflectApply(StringPrototypeSubstring, str, [last]);
-        return result;
-      }
-
-      result += ReflectApply(StringPrototypeSubstring, str, [last, match.index]);
-      result += ReflectApply(replacement, null, match);
-      last = match.index + match[0].length;
-    }
-  }
-
   function StringSplit(str, delimiter) {
     assertEq(typeof str === "string" && typeof delimiter === "string", true,
              "StringSplit must be called with two string arguments");
     assertEq(delimiter.length > 0, true,
              "StringSplit doesn't support an empty delimiter string");
 
     var parts = [];
     var last = 0;
@@ -619,32 +597,20 @@
       } else {
         reportFailure(description + " : " + output);
       }
     }
     return testcase.passed;
   }
   global.reportMatch = reportMatch;
 
-  function normalizeSource(source) {
-    source = String(source);
-    source = StringReplace(source, /([(){},.:\[\]])/mg, (_, punctuator) => ` ${punctuator} `);
-    source = StringReplace(source, /(\w+)/mg, (_, word) => ` ${word} `);
-    source = StringReplace(source, /<(\/)? (\w+) (\/)?>/mg,
-                           (_, left = "", tagName, right = "") => `<${left}${tagName}${right}>`);
-    source = StringReplace(source, /\s+/mg, _ => ' ');
-    source = StringReplace(source, /new (\w+)\s*\(\s*\)/mg, (_, name) => `new ${name}`);
-
-    return source;
-  }
-
   function compareSource(expect, actual, summary) {
     // compare source
-    var expectP = normalizeSource(expect);
-    var actualP = normalizeSource(actual);
+    var expectP = String(expect);
+    var actualP = String(actual);
 
     print('expect:\n' + expectP);
     print('actual:\n' + actualP);
 
     reportCompare(expectP, actualP, summary);
 
     // actual must be compilable if expect is?
     try {
--- a/js/src/vm/TraceLogging.cpp
+++ b/js/src/vm/TraceLogging.cpp
@@ -850,17 +850,17 @@ TraceLoggerThreadState::init()
     const char* options = getenv("TLOPTIONS");
     if (options) {
         if (strstr(options, "help")) {
             fflush(nullptr);
             printf(
                 "\n"
                 "usage: TLOPTIONS=option,option,option,... where options can be:\n"
                 "\n"
-                "  EnableCooperatingThread Start logging cooperating threads immediately.\n"
+                "  EnableActiveThread      Start logging cooperating threads immediately.\n"
                 "  EnableOffThread         Start logging helper threads immediately.\n"
                 "  EnableGraph             Enable spewing the tracelogging graph to a file.\n"
                 "  Errors                  Report errors during tracing to stderr.\n"
             );
             printf("\n");
             exit(0);
             /*NOTREACHED*/
         }
--- a/layout/generic/nsTextFrame.h
+++ b/layout/generic/nsTextFrame.h
@@ -92,16 +92,19 @@ public:
                    GetType() == aNextContinuation->GetType(),
                  "setting a next continuation with incorrect type!");
     NS_ASSERTION(
       !nsSplittableFrame::IsInNextContinuationChain(aNextContinuation, this),
       "creating a loop in continuation chain!");
     mNextContinuation = static_cast<nsTextFrame*>(aNextContinuation);
     if (aNextContinuation)
       aNextContinuation->RemoveStateBits(NS_FRAME_IS_FLUID_CONTINUATION);
+    // Setting a non-fluid continuation might affect our flow length (they're
+    // quite rare so we assume it always does) so we delete our cached value:
+    GetContent()->DeleteProperty(nsGkAtoms::flowlength);
   }
   nsIFrame* GetNextInFlowVirtual() const override { return GetNextInFlow(); }
   nsTextFrame* GetNextInFlow() const
   {
     return mNextContinuation &&
                (mNextContinuation->GetStateBits() &
                 NS_FRAME_IS_FLUID_CONTINUATION)
              ? mNextContinuation
@@ -110,18 +113,25 @@ public:
   void SetNextInFlow(nsIFrame* aNextInFlow) final
   {
     NS_ASSERTION(!aNextInFlow || GetType() == aNextInFlow->GetType(),
                  "setting a next in flow with incorrect type!");
     NS_ASSERTION(
       !nsSplittableFrame::IsInNextContinuationChain(aNextInFlow, this),
       "creating a loop in continuation chain!");
     mNextContinuation = static_cast<nsTextFrame*>(aNextInFlow);
-    if (aNextInFlow)
+    if (mNextContinuation &&
+        !mNextContinuation->HasAnyStateBits(NS_FRAME_IS_FLUID_CONTINUATION)) {
+      // Changing from non-fluid to fluid continuation might affect our flow
+      // length, so we delete our cached value:
+      GetContent()->DeleteProperty(nsGkAtoms::flowlength);
+    }
+    if (aNextInFlow) {
       aNextInFlow->AddStateBits(NS_FRAME_IS_FLUID_CONTINUATION);
+    }
   }
   nsTextFrame* LastInFlow() const final;
   nsTextFrame* LastContinuation() const final;
 
   nsSplittableType GetSplittableType() const final
   {
     return NS_FRAME_SPLITTABLE;
   }
--- a/modules/brotli/moz.build
+++ b/modules/brotli/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Core', 'General')
+
 EXPORTS += [
     'dec/bit_reader.h',
     'dec/decode.h',
     'dec/huffman.h',
     'dec/port.h',
     'dec/state.h',
     'dec/types.h',
 ]
--- a/modules/fdlibm/moz.build
+++ b/modules/fdlibm/moz.build
@@ -1,7 +1,10 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Core', 'JavaScript Engine')
+
 DIRS += ['src']
--- a/modules/libbz2/moz.build
+++ b/modules/libbz2/moz.build
@@ -1,8 +1,11 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Core', 'Build Config')
+
 DIRS += ['src']
 
--- a/modules/libjar/moz.build
+++ b/modules/libjar/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Core', 'Networking: JAR')
+
 if CONFIG['MOZ_ZIPWRITER']:
     DIRS += ['zipwriter']
 
 MOCHITEST_CHROME_MANIFESTS += ['test/chrome/chrome.ini']
 
 MOCHITEST_MANIFESTS += ['test/mochitest/mochitest.ini']
 
 XPCSHELL_TESTS_MANIFESTS += ['test/unit/xpcshell.ini']
--- a/modules/libmar/moz.build
+++ b/modules/libmar/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Toolkit', 'Application Update')
+
 DIRS += ['src']
 
 if CONFIG['MOZ_ENABLE_SIGNMAR']:
     DIRS += ['sign', 'verify']
     TEST_DIRS += ['tests']
 elif CONFIG['MOZ_VERIFY_MAR_SIGNATURE']:
     DIRS += ['verify']
 
--- a/modules/libpref/moz.build
+++ b/modules/libpref/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Core', 'Preferences: Backend')
+
 XPCSHELL_TESTS_MANIFESTS += [
     'test/unit/xpcshell.ini',
     'test/unit_ipc/xpcshell.ini',
 ]
 
 XPIDL_SOURCES += [
     'nsIPrefBranch.idl',
     'nsIPrefBranch2.idl',
new file mode 100644
--- /dev/null
+++ b/modules/moz.build
@@ -0,0 +1,11 @@
+# -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
+# vim: set filetype=python:
+# 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/.
+
+with Files('**'):
+    BUG_COMPONENT = ('Core', 'General')
+
+with Files('freetype2/**'):
+    BUG_COMPONENT = ('Core', 'Graphics: Text')
--- a/modules/woff2/moz.build
+++ b/modules/woff2/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Core', 'Graphics: Text')
+
 EXPORTS += [
     'src/woff2_dec.h',
     'src/woff2_out.h',
 ]
 
 UNIFIED_SOURCES += [
     'src/table_tags.cc',
     'src/variable_length.cc',
--- a/modules/xz-embedded/moz.build
+++ b/modules/xz-embedded/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Firefox for Android', 'General')
+
 EXPORTS += [
     'src/xz.h',
 ]
 
 UNIFIED_SOURCES += [
     'src/xz_crc32.c',
     'src/xz_crc64.c',
     'src/xz_dec_bcj.c',
--- a/modules/zlib/moz.build
+++ b/modules/zlib/moz.build
@@ -1,8 +1,11 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Core', 'General')
+
 DIRS += ['src']
 
--- a/mozglue/moz.build
+++ b/mozglue/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files("**"):
+    BUG_COMPONENT = ("Core", "mozglue")
+
 if CONFIG['MOZ_LINKER']:
     DIRS += ['linker']
 
 if CONFIG['MOZ_WIDGET_TOOLKIT'] == 'android':
     DIRS += ['android']
 
 DIRS += [
   'build',
--- a/netwerk/cache2/CacheFile.cpp
+++ b/netwerk/cache2/CacheFile.cpp
@@ -880,16 +880,18 @@ CacheFile::OpenOutputStream(CacheOutputC
       return NS_ERROR_NOT_AVAILABLE;
     }
   }
 
   if (mAltDataOffset != -1) {
     // Remove alt-data
     rv = Truncate(mAltDataOffset);
     if (NS_FAILED(rv)) {
+      LOG(("CacheFile::OpenOutputStream() - Truncating alt-data failed "
+           "[rv=0x%08" PRIx32 "]", static_cast<uint32_t>(rv)));
       return rv;
     }
     mMetadata->SetElement(CacheFileUtils::kAltDataKey, nullptr);
     mAltDataOffset = -1;
   }
 
   // Once we open output stream we no longer allow preloading of chunks without
   // input stream. There is no reason to believe that some input stream will be
@@ -938,16 +940,18 @@ CacheFile::OpenAlternativeOutputStream(C
       return NS_ERROR_NOT_AVAILABLE;
     }
   }
 
   if (mAltDataOffset != -1) {
     // Truncate old alt-data
     rv = Truncate(mAltDataOffset);
     if (NS_FAILED(rv)) {
+      LOG(("CacheFile::OpenAlternativeOutputStream() - Truncating old alt-data "
+           "failed [rv=0x%08" PRIx32 "]", static_cast<uint32_t>(rv)));
       return rv;
     }
   } else {
     mAltDataOffset = mDataSize;
   }
 
   nsAutoCString altMetadata;
   CacheFileUtils::BuildAlternativeDataInfo(aAltDataType, mAltDataOffset,
@@ -1793,66 +1797,108 @@ CacheFile::BytesFromChunk(uint32_t aInde
   return std::min(advance, tail);
 }
 
 nsresult
 CacheFile::Truncate(int64_t aOffset)
 {
   AssertOwnsLock();
 
+  LOG(("CacheFile::Truncate() [this=%p, offset=%" PRId64 "]", this, aOffset));
+
   nsresult rv;
 
   MOZ_ASSERT(aOffset <= mDataSize);
   MOZ_ASSERT(mReady);
 
   uint32_t lastChunk = 0;
-
-  if (aOffset > 0) {
+  if (mDataSize > 0) {
     lastChunk = (mDataSize - 1) / kChunkSize;
   }
 
-  uint32_t bytesInLastChunk = aOffset - lastChunk * kChunkSize;
-
+  uint32_t newLastChunk = 0;
+  if (aOffset > 0) {
+    newLastChunk = (aOffset - 1) / kChunkSize;
+  }
+
+  uint32_t bytesInNewLastChunk = aOffset - newLastChunk * kChunkSize;
+
+  // Remove all truncated chunks from mCachedChunks
   for (auto iter = mCachedChunks.Iter(); !iter.Done(); iter.Next()) {
     uint32_t idx = iter.Key();
 
-    if (idx > lastChunk) {
+    if (idx > newLastChunk) {
       // This is unused chunk, simply remove it.
+      LOG(("CacheFile::Truncate() - removing cached chunk [idx=%u]", idx));
       iter.Remove();
-      continue;
+    }
+  }
+
+  // Discard all truncated chunks in mChunks
+  for (auto iter = mChunks.Iter(); !iter.Done(); iter.Next()) {
+    uint32_t idx = iter.Key();
+
+    if (idx > newLastChunk) {
+      RefPtr<CacheFileChunk>& chunk = iter.Data();
+      LOG(("CacheFile::Truncate() - discarding chunk [idx=%u, chunk=%p]",
+           idx, chunk.get()));
+      chunk->mDiscardedChunk = true;
+      mDiscardedChunks.AppendElement(chunk);
+      iter.Remove();
     }
-
-    if (idx == lastChunk) {
-      RefPtr<CacheFileChunk>& chunk = iter.Data();
-
-      rv = chunk->Truncate(bytesInLastChunk);
+  }
+
+  // Remove hashes of all removed chunks from the metadata
+  for (uint32_t i = lastChunk; i > newLastChunk; --i) {
+    mMetadata->RemoveHash(i);
+  }
+
+  // Truncate new last chunk
+  if (bytesInNewLastChunk == kChunkSize) {
+    LOG(("CacheFile::Truncate() - not truncating last chunk."));
+  } else {
+    RefPtr<CacheFileChunk> chunk;
+    if (mChunks.Get(newLastChunk, getter_AddRefs(chunk))) {
+      LOG(("CacheFile::Truncate() - New last chunk %p got from mChunks.",
+           chunk.get()));
+    } else if (mCachedChunks.Get(newLastChunk, getter_AddRefs(chunk))) {
+      LOG(("CacheFile::Truncate() - New last chunk %p got from mCachedChunks.",
+           chunk.get()));
+    } else {
+      // New last chunk isn't loaded but we need to update the hash.
+      MOZ_ASSERT(!mMemoryOnly);
+      MOZ_ASSERT(mHandle);
+
+      rv = GetChunkLocked(newLastChunk, PRELOADER, nullptr,
+                          getter_AddRefs(chunk));
       if (NS_FAILED(rv)) {
         return rv;
       }
+      // We've checked that we don't have this chunk, so no chunk must be
+      // returned.
+      MOZ_ASSERT(!chunk);
+
+      if (!mChunks.Get(newLastChunk, getter_AddRefs(chunk))) {
+        return NS_ERROR_UNEXPECTED;
+      }
+
+      LOG(("CacheFile::Truncate() - New last chunk %p got from preloader.",
+           chunk.get()));
     }
-  }
-
-  for (auto iter = mChunks.Iter(); !iter.Done(); iter.Next()) {
-    uint32_t idx = iter.Key();
-    RefPtr<CacheFileChunk>& chunk = iter.Data();
-
-    if (idx > lastChunk) {
-      chunk->mDiscardedChunk = true;
-      mDiscardedChunks.AppendElement(chunk);
-      iter.Remove();
-      continue;
+
+    rv = chunk->Truncate(bytesInNewLastChunk);
+    if (NS_FAILED(rv)) {
+      return rv;
     }
 
-    if (idx == lastChunk) {
-      RefPtr<CacheFileChunk>& chunk = iter.Data();
-
-      rv = chunk->Truncate(bytesInLastChunk);
-      if (NS_FAILED(rv)) {
-        return rv;
-      }
+    // If the chunk is ready set the new hash now. If it's still being loaded
+    // CacheChunk::Truncate() made the chunk dirty and the hash will be updated
+    // in OnChunkWritten().
+    if (chunk->IsReady()) {
+      mMetadata->SetHash(newLastChunk, chunk->Hash());
     }
   }
 
   if (mHandle) {
     rv = CacheFileIOManager::TruncateSeekSetEOF(mHandle, aOffset, aOffset, nullptr);
     if (NS_FAILED(rv)) {
       return rv;
     }
--- a/netwerk/cache2/CacheFileChunk.cpp
+++ b/netwerk/cache2/CacheFileChunk.cpp
@@ -123,17 +123,17 @@ CacheFileChunkBuffer::SetDataSize(uint32
     // EnsureBufSize must be called before SetDataSize, so the new data size
     // is guaranteed to be smaller than or equal to mBufSize.
     aDataSize <= mBufSize ||
     // The only exception is an optimization when we read the data from the
     // disk. The data is read to a separate buffer and CacheFileChunk::mBuf is
     // empty (see CacheFileChunk::Read). We need to set mBuf::mDataSize
     // accordingly so that DataSize() methods return correct value, but we don't
     // want to allocate the buffer since it wouldn't be used in most cases.
-    (mDataSize == 0 && mBufSize == 0 && mChunk->mState == CacheFileChunk::READING));
+    (mBufSize == 0 && mChunk->mState == CacheFileChunk::READING));
 
   mDataSize = aDataSize;
 }
 
 void
 CacheFileChunkBuffer::AssertOwnsLock() const
 {
   mChunk->AssertOwnsLock();
@@ -560,18 +560,16 @@ uint32_t
 CacheFileChunk::Index() const
 {
   return mIndex;
 }
 
 CacheHash::Hash16_t
 CacheFileChunk::Hash() const
 {
-  AssertOwnsLock();
-
   MOZ_ASSERT(!mListener);
   MOZ_ASSERT(IsReady());
 
   return CacheHash::Hash16(mBuf->Buf(), mBuf->DataSize());
 }
 
 uint32_t
 CacheFileChunk::DataSize() const
@@ -621,16 +619,22 @@ CacheFileChunk::UpdateDataSize(uint32_t 
 
   mValidityMap.AddPair(aOffset, aLen);
   mValidityMap.Log();
 }
 
 nsresult
 CacheFileChunk::Truncate(uint32_t aOffset)
 {
+  MOZ_RELEASE_ASSERT(mState == READY || mState == WRITING || mState == READING);
+
+  if (mState == READING) {
+    mIsDirty = true;
+  }
+
   mBuf->SetDataSize(aOffset);
   return NS_OK;
 }
 
 nsresult
 CacheFileChunk::OnFileOpened(CacheFileHandle *aHandle, nsresult aResult)
 {
   MOZ_CRASH("CacheFileChunk::OnFileOpened should not be called!");
@@ -692,16 +696,21 @@ CacheFileChunk::OnDataRead(CacheFileHand
       CacheHash::Hash16_t hash = CacheHash::Hash16(tmpBuf->Buf(),
                                                    tmpBuf->DataSize());
       if (hash != mExpectedHash) {
         LOG(("CacheFileChunk::OnDataRead() - Hash mismatch! Hash of the data is"
              " %hx, hash in metadata is %hx. [this=%p, idx=%d]",
              hash, mExpectedHash, this, mIndex));
         aResult = NS_ERROR_FILE_CORRUPTED;
       } else {
+        if (mBuf->DataSize() < tmpBuf->DataSize()) {
+          // Truncate() was called while the data was being read.
+          tmpBuf->SetDataSize(mBuf->DataSize());
+        }
+
         if (!mBuf->Buf()) {
           // Just swap the buffers if mBuf is still empty
           mBuf.swap(tmpBuf);
         } else {
           LOG(("CacheFileChunk::OnDataRead() - Merging buffers. [this=%p]",
                this));
 
           mValidityMap.Log();
@@ -751,18 +760,16 @@ bool
 CacheFileChunk::IsKilled()
 {
   return mFile->IsKilled();
 }
 
 bool
 CacheFileChunk::IsReady() const
 {
-  AssertOwnsLock();
-
   return (NS_SUCCEEDED(mStatus) && (mState == READY || mState == WRITING));
 }
 
 bool
 CacheFileChunk::IsDirty() const
 {
   AssertOwnsLock();
 
--- a/netwerk/cache2/CacheFileMetadata.cpp
+++ b/netwerk/cache2/CacheFileMetadata.cpp
@@ -529,16 +529,33 @@ CacheFileMetadata::SetHash(uint32_t aInd
   NetworkEndian::writeUint16(&mHashArray[aIndex], aHash);
 
   DoMemoryReport(MemoryUsage());
 
   return NS_OK;
 }
 
 nsresult
+CacheFileMetadata::RemoveHash(uint32_t aIndex)
+{
+  LOG(("CacheFileMetadata::RemoveHash() [this=%p, idx=%d]", this, aIndex));
+
+  MarkDirty();
+
+  MOZ_ASSERT((aIndex + 1) == mHashCount, "Can remove only last hash!");
+
+  if (aIndex + 1 != mHashCount) {
+    return NS_ERROR_INVALID_ARG;
+  }
+
+  mHashCount--;
+  return NS_OK;
+}
+
+nsresult
 CacheFileMetadata::AddFlags(uint32_t aFlags)
 {
   MarkDirty(false);
   mMetaHdr.mFlags |= aFlags;
   return NS_OK;
 }
 
 nsresult
--- a/netwerk/cache2/CacheFileMetadata.h
+++ b/netwerk/cache2/CacheFileMetadata.h
@@ -151,16 +151,17 @@ public:
   bool     Pinned() const      { return !!(mMetaHdr.mFlags & kCacheEntryIsPinned); }
 
   const char * GetElement(const char *aKey);
   nsresult     SetElement(const char *aKey, const char *aValue);
   nsresult     Visit(nsICacheEntryMetaDataVisitor *aVisitor);
 
   CacheHash::Hash16_t GetHash(uint32_t aIndex);
   nsresult            SetHash(uint32_t aIndex, CacheHash::Hash16_t aHash);
+  nsresult            RemoveHash(uint32_t aIndex);
 
   nsresult AddFlags(uint32_t aFlags);
   nsresult RemoveFlags(uint32_t aFlags);
   nsresult GetFlags(uint32_t *_retval);
   nsresult SetExpirationTime(uint32_t aExpirationTime);
   nsresult GetExpirationTime(uint32_t *_retval);
   nsresult SetFrecency(uint32_t aFrecency);
   nsresult GetFrecency(uint32_t *_retval);
--- a/netwerk/ipc/NeckoChannelParams.ipdlh
+++ b/netwerk/ipc/NeckoChannelParams.ipdlh
@@ -7,17 +7,16 @@
 
 include protocol PHttpChannel;
 include protocol PFTPChannel;
 include protocol PRtspChannel;
 include protocol PSendStream;
 include BlobTypes;
 include URIParams;
 include IPCStream;
-include InputStreamParams;
 include PBackgroundSharedTypes;
 
 using mozilla::OriginAttributes from "mozilla/ipc/BackgroundUtils.h";
 using struct mozilla::void_t from "ipc/IPCMessageUtils.h";
 using RequestHeaderTuples from "mozilla/net/PHttpChannelParams.h";
 using struct nsHttpAtom from "nsHttp.h";
 using class nsHttpResponseHead from "nsHttpResponseHead.h";
 
@@ -127,16 +126,17 @@ struct HttpChannelOpenArgs
   uint32_t                    initialRwin;
   bool                        blockAuthPrompt;
   bool                        suspendAfterSynthesizeResponse;
   bool                        allowStaleCacheContent;
   nsCString                   contentTypeHint;
   nsCString                   channelId;
   uint64_t                    contentWindowId;
   nsCString                   preferredAlternativeType;
+  uint64_t                    topLevelOuterContentWindowId;
 };
 
 struct HttpChannelConnectArgs
 {
   uint32_t registrarId;
   bool shouldIntercept;
 };
 
@@ -150,17 +150,17 @@ union HttpChannelCreationArgs
 // FTP IPDL structs
 //-----------------------------------------------------------------------------
 
 struct FTPChannelOpenArgs
 {
   URIParams                 uri;
   uint64_t                  startPos;
   nsCString                 entityID;
-  OptionalInputStreamParams uploadStream;
+  OptionalIPCStream         uploadStream;
   OptionalLoadInfoArgs      loadInfo;
 };
 
 struct FTPChannelConnectArgs
 {
   uint32_t channelId;
 };
 
--- a/netwerk/ipc/PNecko.ipdl
+++ b/netwerk/ipc/PNecko.ipdl
@@ -22,17 +22,16 @@ include protocol PBlob; //FIXME: bug #79
 include protocol PFileDescriptorSet;
 include protocol PDataChannel;
 include protocol PTransportProvider;
 include protocol PSendStream;
 
 include protocol PRtspController;
 include protocol PRtspChannel;
 include URIParams;
-include InputStreamParams;
 include NeckoChannelParams;
 include PBrowserOrId;
 include protocol PAltDataOutputStream;
 
 using class IPC::SerializedLoadContext from "SerializedLoadContext.h";
 using mozilla::dom::TabId from "mozilla/dom/ipc/IdType.h";
 using class IPC::Principal from "mozilla/dom/PermissionMessageUtils.h";
 
--- a/netwerk/protocol/ftp/FTPChannelChild.cpp
+++ b/netwerk/protocol/ftp/FTPChannelChild.cpp
@@ -3,23 +3,24 @@
 
 /* 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 "mozilla/net/NeckoChild.h"
 #include "mozilla/net/ChannelDiverterChild.h"
 #include "mozilla/net/FTPChannelChild.h"
+#include "mozilla/dom/ContentChild.h"
 #include "mozilla/dom/TabChild.h"
 #include "nsFtpProtocolHandler.h"
 #include "nsITabChild.h"
 #include "nsStringStream.h"
 #include "nsNetUtil.h"
 #include "base/compiler_specific.h"
-#include "mozilla/ipc/InputStreamUtils.h"
+#include "mozilla/ipc/IPCStreamUtils.h"
 #include "mozilla/ipc/URIUtils.h"
 #include "SerializedLoadContext.h"
 #include "mozilla/ipc/BackgroundUtils.h"
 #include "nsIPrompt.h"
 
 using namespace mozilla::ipc;
 
 #undef LOG
@@ -177,27 +178,25 @@ FTPChannelChild::AsyncOpen(::nsIStreamLi
 
   mListener = listener;
   mListenerContext = aContext;
 
   // add ourselves to the load group. 
   if (mLoadGroup)
     mLoadGroup->AddRequest(this, nullptr);
 
-  OptionalInputStreamParams uploadStream;
-  nsTArray<mozilla::ipc::FileDescriptor> fds;
-  SerializeInputStream(mUploadStream, uploadStream, fds);
-
-  MOZ_ASSERT(fds.IsEmpty());
+  mozilla::ipc::AutoIPCStream autoStream;
+  autoStream.Serialize(mUploadStream,
+                       static_cast<mozilla::dom::ContentChild*>(gNeckoChild->Manager()));
 
   FTPChannelOpenArgs openArgs;
   SerializeURI(nsBaseChannel::URI(), openArgs.uri());
   openArgs.startPos() = mStartPos;
   openArgs.entityID() = mEntityID;
-  openArgs.uploadStream() = uploadStream;
+  openArgs.uploadStream() = autoStream.TakeOptionalValue();
 
   nsCOMPtr<nsILoadInfo> loadInfo;
   GetLoadInfo(getter_AddRefs(loadInfo));
   rv = mozilla::ipc::LoadInfoToLoadInfoArgs(loadInfo, &openArgs.loadInfo());
   NS_ENSURE_SUCCESS(rv, rv);
 
   gNeckoChild->
     SendPFTPChannelConstructor(this, tabChild, IPC::SerializedLoadContext(this),
--- a/netwerk/protocol/ftp/FTPChannelParent.cpp
+++ b/netwerk/protocol/ftp/FTPChannelParent.cpp
@@ -14,17 +14,17 @@
 #include "nsNetUtil.h"
 #include "nsQueryObject.h"
 #include "nsFtpProtocolHandler.h"
 #include "nsIAuthPrompt.h"
 #include "nsIAuthPromptProvider.h"
 #include "nsIEncodedChannel.h"
 #include "nsIHttpChannelInternal.h"
 #include "nsIForcePendingChannel.h"
-#include "mozilla/ipc/InputStreamUtils.h"
+#include "mozilla/ipc/IPCStreamUtils.h"
 #include "mozilla/ipc/URIUtils.h"
 #include "mozilla/Unused.h"
 #include "SerializedLoadContext.h"
 #include "nsIContentPolicy.h"
 #include "mozilla/ipc/BackgroundUtils.h"
 #include "mozilla/LoadInfo.h"
 
 using namespace mozilla::dom;
@@ -112,17 +112,17 @@ FTPChannelParent::Init(const FTPChannelC
     return false;
   }
 }
 
 bool
 FTPChannelParent::DoAsyncOpen(const URIParams& aURI,
                               const uint64_t& aStartPos,
                               const nsCString& aEntityID,
-                              const OptionalInputStreamParams& aUploadStream,
+                              const OptionalIPCStream& aUploadStream,
                               const OptionalLoadInfoArgs& aLoadInfoArgs)
 {
   nsresult rv;
 
   nsCOMPtr<nsIURI> uri = DeserializeURI(aURI);
   if (!uri)
       return false;
 
@@ -165,18 +165,17 @@ FTPChannelParent::DoAsyncOpen(const URIP
 
   if (mPBOverride != kPBOverride_Unset) {
     ftpChan->SetPrivate(mPBOverride == kPBOverride_Private ? true : false);
   }
   rv = ftpChan->SetNotificationCallbacks(this);
   if (NS_FAILED(rv))
     return SendFailedAsyncOpen(rv);
 
-  nsTArray<mozilla::ipc::FileDescriptor> fds;
-  nsCOMPtr<nsIInputStream> upload = DeserializeInputStream(aUploadStream, fds);
+  nsCOMPtr<nsIInputStream> upload = DeserializeIPCStream(aUploadStream);
   if (upload) {
     // contentType and contentLength are ignored
     rv = ftpChan->SetUploadStream(upload, EmptyCString(), 0);
     if (NS_FAILED(rv))
       return SendFailedAsyncOpen(rv);
   }
 
   rv = ftpChan->ResumeAt(aStartPos, aEntityID);
--- a/netwerk/protocol/ftp/FTPChannelParent.h
+++ b/netwerk/protocol/ftp/FTPChannelParent.h
@@ -72,17 +72,17 @@ protected:
   // private, supporting function for ADivertableParentChannel.
   nsresult ResumeForDiversion();
 
   // Asynchronously calls NotifyDiversionFailed.
   void FailDiversion(nsresult aErrorCode, bool aSkipResume = true);
 
   bool DoAsyncOpen(const URIParams& aURI, const uint64_t& aStartPos,
                    const nsCString& aEntityID,
-                   const OptionalInputStreamParams& aUploadStream,
+                   const OptionalIPCStream& aUploadStream,
                    const OptionalLoadInfoArgs& aLoadInfoArgs);
 
   // used to connect redirected-to channel in parent with just created
   // ChildChannel.  Used during HTTP->FTP redirects.
   bool ConnectChannel(const uint32_t& channelId);
 
   void DivertOnDataAvailable(const nsCString& data,
                              const uint64_t& offset,
--- a/netwerk/protocol/ftp/PFTPChannel.ipdl
+++ b/netwerk/protocol/ftp/PFTPChannel.ipdl
@@ -1,17 +1,16 @@
 /* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
 /* vim: set sw=2 ts=8 et tw=80 ft=cpp : */
 
 /* 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 protocol PNecko;
-include InputStreamParams;
 include URIParams;
 
 //FIXME: bug #792908 (NeckoChannelParams already included by PNecko)
 include NeckoChannelParams;
 include protocol PBlob; //FIXME: bug #792908
 
 using PRTime from "prtime.h";
 
--- a/netwerk/protocol/http/HttpBaseChannel.cpp
+++ b/netwerk/protocol/http/HttpBaseChannel.cpp
@@ -195,16 +195,17 @@ HttpBaseChannel::HttpBaseChannel()
   , mFetchCacheMode(nsIHttpChannelInternal::FETCH_CACHE_MODE_DEFAULT)
   , mOnStartRequestCalled(false)
   , mOnStopRequestCalled(false)
   , mAfterOnStartRequestBegun(false)
   , mTransferSize(0)
   , mDecodedBodySize(0)
   , mEncodedBodySize(0)
   , mContentWindowId(0)
+  , mTopLevelOuterContentWindowId(0)
   , mRequireCORSPreflight(false)
   , mReportCollector(new ConsoleReportCollector())
   , mForceMainDocumentChannel(false)
   , mIsTrackingResource(false)
 {
   LOG(("Creating HttpBaseChannel @%p\n", this));
 
   // Subfields of unions cannot be targeted in an initializer list.
@@ -3826,16 +3827,39 @@ HttpBaseChannel::EnsureRequestContextID(
     }
 
     // Set the load group connection scope on the transaction
     rootLoadGroup->GetRequestContextID(&mRequestContextID);
     return true;
 }
 
 void
+HttpBaseChannel::EnsureTopLevelOuterContentWindowId()
+{
+  if (mTopLevelOuterContentWindowId) {
+    return;
+  }
+
+  nsCOMPtr<nsILoadContext> loadContext;
+  GetCallback(loadContext);
+  if (!loadContext) {
+    return;
+  }
+
+  nsCOMPtr<mozIDOMWindowProxy> topWindow;
+  loadContext->GetTopWindow(getter_AddRefs(topWindow));
+  if (!topWindow) {
+    return;
+  }
+
+  mTopLevelOuterContentWindowId =
+    nsPIDOMWindowOuter::From(topWindow)->WindowID();
+}
+
+void
 HttpBaseChannel::SetCorsPreflightParameters(const nsTArray<nsCString>& aUnsafeHeaders)
 {
   MOZ_RELEASE_ASSERT(!mRequestObserversCalled);
 
   mRequireCORSPreflight = true;
   mUnsafeHeaders = aUnsafeHeaders;
 }
 
--- a/netwerk/protocol/http/HttpBaseChannel.h
+++ b/netwerk/protocol/http/HttpBaseChannel.h
@@ -353,16 +353,21 @@ public: /* Necko internal use only... */
     // the new mUploadStream.
     void EnsureUploadStreamIsCloneableComplete(nsresult aStatus);
 
     void SetIsTrackingResource()
     {
       mIsTrackingResource = true;
     }
 
+    void SetTopLevelOuterContentWindowId(uint64_t aTopLevelOuterContentWindowId)
+    {
+      mTopLevelOuterContentWindowId = aTopLevelOuterContentWindowId;
+    }
+
 protected:
   // Handle notifying listener, removing from loadgroup if request failed.
   void     DoNotifyListener();
   virtual void DoNotifyListenerCleanup() = 0;
 
   // drop reference to listener, its callbacks, and the progress sink
   void ReleaseListeners();
 
@@ -582,16 +587,19 @@ protected:
 
   nsID mRequestContextID;
   bool EnsureRequestContextID();
 
   // ID of the top-level document's inner window this channel is being
   // originated from.
   uint64_t mContentWindowId;
 
+  uint64_t mTopLevelOuterContentWindowId;
+  void EnsureTopLevelOuterContentWindowId();
+
   bool                              mRequireCORSPreflight;
   nsTArray<nsCString>               mUnsafeHeaders;
 
   nsCOMPtr<nsIConsoleReportCollector> mReportCollector;
 
   // Holds the name of the preferred alt-data type.
   nsCString mPreferredCachedAltDataType;
   // Holds the name of the alternative data type the channel returned.
--- a/netwerk/protocol/http/HttpChannelChild.cpp
+++ b/netwerk/protocol/http/HttpChannelChild.cpp
@@ -2142,16 +2142,17 @@ HttpChannelChild::ContinueAsyncOpen()
   // This id identifies the inner window's top-level document,
   // which changes on every new load or navigation.
   uint64_t contentWindowId = 0;
   if (tabChild) {
     MOZ_ASSERT(tabChild->WebNavigation());
     nsCOMPtr<nsIDocument> document = tabChild->GetDocument();
     if (document) {
       contentWindowId = document->InnerWindowID();
+      mTopLevelOuterContentWindowId = document->OuterWindowID();
     }
   }
   SetTopLevelContentWindowId(contentWindowId);
 
   HttpChannelOpenArgs openArgs;
   // No access to HttpChannelOpenArgs members, but they each have a
   // function with the struct name that returns a ref.
   SerializeURI(mURI, openArgs.uri());
@@ -2240,16 +2241,17 @@ HttpChannelChild::ContinueAsyncOpen()
   mRequestContextID.ToProvidedString(rcid);
   openArgs.requestContextID().AssignASCII(rcid);
 
   char chid[NSID_LENGTH];
   mChannelId.ToProvidedString(chid);
   openArgs.channelId().AssignASCII(chid);
 
   openArgs.contentWindowId() = contentWindowId;
+  openArgs.topLevelOuterContentWindowId() = mTopLevelOuterContentWindowId;
 
   if (tabChild && !tabChild->IPCOpen()) {
     return NS_ERROR_FAILURE;
   }
 
   ContentChild* cc = static_cast<ContentChild*>(gNeckoChild->Manager());
   if (cc->IsShuttingDown()) {
     return NS_ERROR_FAILURE;
--- a/netwerk/protocol/http/HttpChannelParent.cpp
+++ b/netwerk/protocol/http/HttpChannelParent.cpp
@@ -125,17 +125,18 @@ HttpChannelParent::Init(const HttpChanne
                        a.entityID(), a.chooseApplicationCache(),
                        a.appCacheClientID(), a.allowSpdy(), a.allowAltSvc(), a.beConservative(),
                        a.loadInfo(), a.synthesizedResponseHead(),
                        a.synthesizedSecurityInfoSerialization(),
                        a.cacheKey(), a.requestContextID(), a.preflightArgs(),
                        a.initialRwin(), a.blockAuthPrompt(),
                        a.suspendAfterSynthesizeResponse(),
                        a.allowStaleCacheContent(), a.contentTypeHint(),
-                       a.channelId(), a.contentWindowId(), a.preferredAlternativeType());
+                       a.channelId(), a.contentWindowId(), a.preferredAlternativeType(),
+                       a.topLevelOuterContentWindowId());
   }
   case HttpChannelCreationArgs::THttpChannelConnectArgs:
   {
     const HttpChannelConnectArgs& cArgs = aArgs.get_HttpChannelConnectArgs();
     return ConnectChannel(cArgs.registrarId(), cArgs.shouldIntercept());
   }
   default:
     NS_NOTREACHED("unknown open type");
@@ -323,17 +324,18 @@ HttpChannelParent::DoAsyncOpen(  const U
                                  const OptionalCorsPreflightArgs& aCorsPreflightArgs,
                                  const uint32_t&            aInitialRwin,
                                  const bool&                aBlockAuthPrompt,
                                  const bool&                aSuspendAfterSynthesizeResponse,
                                  const bool&                aAllowStaleCacheContent,
                                  const nsCString&           aContentTypeHint,
                                  const nsCString&           aChannelId,
                                  const uint64_t&            aContentWindowId,
-                                 const nsCString&           aPreferredAlternativeType)
+                                 const nsCString&           aPreferredAlternativeType,
+                                 const uint64_t&            aTopLevelOuterContentWindowId)
 {
   nsCOMPtr<nsIURI> uri = DeserializeURI(aURI);
   if (!uri) {
     // URIParams does MOZ_ASSERT if null, but we need to protect opt builds from
     // null deref here.
     return false;
   }
   nsCOMPtr<nsIURI> originalUri = DeserializeURI(aOriginalURI);
@@ -374,16 +376,17 @@ HttpChannelParent::DoAsyncOpen(  const U
   mChannel = do_QueryObject(channel, &rv);
   if (NS_FAILED(rv)) {
     return SendFailedAsyncOpen(rv);
   }
 
   // Set the channelId allocated in child to the parent instance
   mChannel->SetChannelId(aChannelId);
   mChannel->SetTopLevelContentWindowId(aContentWindowId);
+  mChannel->SetTopLevelOuterContentWindowId(aTopLevelOuterContentWindowId);
 
   mChannel->SetWarningReporter(this);
   mChannel->SetTimingEnabled(true);
   if (mPBOverride != kPBOverride_Unset) {
     mChannel->SetPrivate(mPBOverride == kPBOverride_Private ? true : false);
   }
 
   if (doResumeAt)
--- a/netwerk/protocol/http/HttpChannelParent.h
+++ b/netwerk/protocol/http/HttpChannelParent.h
@@ -143,17 +143,18 @@ protected:
               const OptionalCorsPreflightArgs& aCorsPreflightArgs,
               const uint32_t&            aInitialRwin,
               const bool&                aBlockAuthPrompt,
               const bool&                aSuspendAfterSynthesizeResponse,
               const bool&                aAllowStaleCacheContent,
               const nsCString&           aContentTypeHint,
               const nsCString&           aChannelId,
               const uint64_t&            aContentWindowId,
-              const nsCString&           aPreferredAlternativeType);
+              const nsCString&           aPreferredAlternativeType,
+              const uint64_t&            aTopLevelOuterContentWindowId);
 
   virtual mozilla::ipc::IPCResult RecvSetPriority(const int16_t& priority) override;
   virtual mozilla::ipc::IPCResult RecvSetClassOfService(const uint32_t& cos) override;
   virtual mozilla::ipc::IPCResult RecvSetCacheTokenCachedCharset(const nsCString& charset) override;
   virtual mozilla::ipc::IPCResult RecvSuspend() override;
   virtual mozilla::ipc::IPCResult RecvResume() override;
   virtual mozilla::ipc::IPCResult RecvCancel(const nsresult& status) override;
   virtual mozilla::ipc::IPCResult RecvRedirect2Verify(const nsresult& result,
--- a/netwerk/protocol/http/nsHttpChannel.cpp
+++ b/netwerk/protocol/http/nsHttpChannel.cpp
@@ -987,20 +987,23 @@ nsHttpChannel::SetupTransaction()
     NS_QueryNotificationCallbacks(mCallbacks,
                                   mLoadGroup,
                                   NS_GET_IID(nsIHttpPushListener),
                                   getter_AddRefs(pushListener));
     if (pushListener) {
         mCaps |= NS_HTTP_ONPUSH_LISTENER;
     }
 
+    EnsureTopLevelOuterContentWindowId();
+
     nsCOMPtr<nsIAsyncInputStream> responseStream;
     rv = mTransaction->Init(mCaps, mConnectionInfo, &mRequestHead,
                             mUploadStream, mUploadStreamHasHeaders,
                             NS_GetCurrentThread(), callbacks, this,
+                            mTopLevelOuterContentWindowId,
                             getter_AddRefs(responseStream));
     if (NS_FAILED(rv)) {
         mTransaction = nullptr;
         return rv;
     }
 
     mTransaction->SetClassOfService(mClassOfService);
     SetupTransactionRequestContext();
--- a/netwerk/protocol/http/nsHttpConnectionMgr.cpp
+++ b/netwerk/protocol/http/nsHttpConnectionMgr.cpp
@@ -82,16 +82,17 @@ nsHttpConnectionMgr::nsHttpConnectionMgr
     , mNumActiveConns(0)
     , mNumIdleConns(0)
     , mNumSpdyActiveConns(0)
     , mNumHalfOpenConns(0)
     , mTimeOfNextWakeUp(UINT64_MAX)
     , mPruningNoTraffic(false)
     , mTimeoutTickArmed(false)
     , mTimeoutTickNext(1)
+    , mCurrentTopLevelOuterContentWindowId(0)
 {
     LOG(("Creating nsHttpConnectionMgr @%p\n", this));
 }
 
 nsHttpConnectionMgr::~nsHttpConnectionMgr()
 {
     LOG(("Destroying nsHttpConnectionMgr @%p\n", this));
     if (mTimeoutTick)
@@ -2423,16 +2424,54 @@ nsHttpConnectionMgr::ActivateTimeoutTick
         mTimeoutTick->SetTarget(mSocketThreadTarget);
     }
 
     MOZ_ASSERT(!mTimeoutTickArmed, "timer tick armed");
     mTimeoutTickArmed = true;
     mTimeoutTick->Init(this, 1000, nsITimer::TYPE_REPEATING_SLACK);
 }
 
+class UINT64Wrapper : public ARefBase
+{
+public:
+    explicit UINT64Wrapper(uint64_t aUint64) : mUint64(aUint64) {}
+    NS_INLINE_DECL_THREADSAFE_REFCOUNTING(UINT64Wrapper)
+
+    uint64_t GetValue()
+    {
+        return mUint64;
+    }
+private:
+    uint64_t mUint64;
+    virtual ~UINT64Wrapper() = default;
+};
+
+nsresult
+nsHttpConnectionMgr::UpdateCurrentTopLevelOuterContentWindowId(
+    uint64_t aWindowId)
+{
+    RefPtr<UINT64Wrapper> windowIdWrapper = new UINT64Wrapper(aWindowId);
+    return PostEvent(
+        &nsHttpConnectionMgr::OnMsgUpdateCurrentTopLevelOuterContentWindowId,
+        0,
+        windowIdWrapper);
+}
+
+void
+nsHttpConnectionMgr::OnMsgUpdateCurrentTopLevelOuterContentWindowId(
+    int32_t, ARefBase *param)
+{
+    MOZ_ASSERT(PR_GetCurrentThread() == gSocketThread);
+    mCurrentTopLevelOuterContentWindowId =
+        static_cast<UINT64Wrapper*>(param)->GetValue();
+    LOG(("nsHttpConnectionMgr::OnMsgUpdateCurrentTopLevelOuterContentWindowId"
+         " id=%" PRIu64 "\n",
+         mCurrentTopLevelOuterContentWindowId));
+}
+
 void
 nsHttpConnectionMgr::TimeoutTick()
 {
     MOZ_ASSERT(PR_GetCurrentThread() == gSocketThread);
     MOZ_ASSERT(mTimeoutTick, "no readtimeout tick");
 
     LOG(("nsHttpConnectionMgr::TimeoutTick active=%d\n", mNumActiveConns));
     // The next tick will be between 1 second and 1 hr
--- a/netwerk/protocol/http/nsHttpConnectionMgr.h
+++ b/netwerk/protocol/http/nsHttpConnectionMgr.h
@@ -187,16 +187,18 @@ public:
 
     void ResetIPFamilyPreference(nsHttpConnectionInfo *);
 
     uint16_t MaxRequestDelay() { return mMaxRequestDelay; }
 
     // public, so that the SPDY/http2 seesions can activate
     void ActivateTimeoutTick();
 
+    nsresult UpdateCurrentTopLevelOuterContentWindowId(uint64_t aWindowId);
+
 private:
     virtual ~nsHttpConnectionMgr();
 
     class nsHalfOpenSocket;
 
     // nsConnectionEntry
     //
     // mCT maps connection info hash key to nsConnectionEntry object, which
@@ -439,16 +441,17 @@ private:
     void OnMsgCompleteUpgrade      (int32_t, ARefBase *);
     void OnMsgUpdateParam          (int32_t, ARefBase *);
     void OnMsgDoShiftReloadConnectionCleanup (int32_t, ARefBase *);
     void OnMsgProcessFeedback      (int32_t, ARefBase *);
     void OnMsgProcessAllSpdyPendingQ (int32_t, ARefBase *);
     void OnMsgUpdateRequestTokenBucket (int32_t, ARefBase *);
     void OnMsgVerifyTraffic (int32_t, ARefBase *);
     void OnMsgPruneNoTraffic (int32_t, ARefBase *);
+    void OnMsgUpdateCurrentTopLevelOuterContentWindowId (int32_t, ARefBase *);
 
     // Total number of active connections in all of the ConnectionEntry objects
     // that are accessed from mCT connection table.
     uint16_t mNumActiveConns;
     // Total number of idle connections in all of the ConnectionEntry objects
     // that are accessed from mCT connection table.
     uint16_t mNumIdleConns;
     // Total number of spdy connections which are a subset of the active conns
@@ -483,14 +486,15 @@ private:
 
     // Read Timeout Tick handlers
     void TimeoutTick();
 
     // For diagnostics
     void OnMsgPrintDiagnostics(int32_t, ARefBase *);
 
     nsCString mLogData;
+    uint64_t mCurrentTopLevelOuterContentWindowId;
 };
 
 } // namespace net
 } // namespace mozilla
 
 #endif // !nsHttpConnectionMgr_h__
--- a/netwerk/protocol/http/nsHttpTransaction.cpp
+++ b/netwerk/protocol/http/nsHttpTransaction.cpp
@@ -130,16 +130,17 @@ nsHttpTransaction::nsHttpTransaction()
     , mContentDecoding(false)
     , mContentDecodingCheck(false)
     , mDeferredSendProgress(false)
     , mWaitingOnPipeOut(false)
     , mReportedStart(false)
     , mReportedResponseHeader(false)
     , mForTakeResponseHead(nullptr)
     , mResponseHeadTaken(false)
+    , mTopLevelOuterContentWindowId(0)
     , mSubmittedRatePacing(false)
     , mPassedRatePacing(false)
     , mSynchronousRatePaceRequest(false)
     , mClassOfService(0)
     , m0RTTInProgress(false)
     , mTransportStatus(NS_OK)
 {
     LOG(("Creating nsHttpTransaction @%p\n", this));
@@ -182,27 +183,30 @@ nsresult
 nsHttpTransaction::Init(uint32_t caps,
                         nsHttpConnectionInfo *cinfo,
                         nsHttpRequestHead *requestHead,
                         nsIInputStream *requestBody,
                         bool requestBodyHasHeaders,
                         nsIEventTarget *target,
                         nsIInterfaceRequestor *callbacks,
                         nsITransportEventSink *eventsink,
+                        uint64_t topLevelOuterContentWindowId,
                         nsIAsyncInputStream **responseBody)
 {
     nsresult rv;
 
     LOG(("nsHttpTransaction::Init [this=%p caps=%x]\n", this, caps));
 
     MOZ_ASSERT(cinfo);
     MOZ_ASSERT(requestHead);
     MOZ_ASSERT(target);
     MOZ_ASSERT(NS_IsMainThread());
 
+    mTopLevelOuterContentWindowId = topLevelOuterContentWindowId;
+
     mActivityDistributor = do_GetService(NS_HTTPACTIVITYDISTRIBUTOR_CONTRACTID, &rv);
     if (NS_FAILED(rv)) return rv;
 
     bool activityDistributorActive;
     rv = mActivityDistributor->GetIsActive(&activityDistributorActive);
     if (NS_SUCCEEDED(rv) && activityDistributorActive) {
         // there are some observers registered at activity distributor, gather
         // nsISupports for the channel that called Init()
--- a/netwerk/protocol/http/nsHttpTransaction.h
+++ b/netwerk/protocol/http/nsHttpTransaction.h
@@ -69,29 +69,33 @@ public:
     // @param reqBody
     //        the request body (POST or PUT data stream)
     // @param reqBodyIncludesHeaders
     //        fun stuff to support NPAPI plugins.
     // @param target
     //        the dispatch target were notifications should be sent.
     // @param callbacks
     //        the notification callbacks to be given to PSM.
+    // @param topLevelOuterContentWindowId
+    //        indicate the top level outer content window in which
+    //        this transaction is being loaded.
     // @param responseBody
     //        the input stream that will contain the response data.  async
     //        wait on this input stream for data.  on first notification,
     //        headers should be available (check transaction status).
     //
     MOZ_MUST_USE nsresult Init(uint32_t               caps,
                                nsHttpConnectionInfo  *connInfo,
                                nsHttpRequestHead     *reqHeaders,
                                nsIInputStream        *reqBody,
                                bool                   reqBodyIncludesHeaders,
                                nsIEventTarget        *consumerTarget,
                                nsIInterfaceRequestor *callbacks,
                                nsITransportEventSink *eventsink,
+                               uint64_t               topLevelOuterContentWindowId,
                                nsIAsyncInputStream  **responseBody);
 
     // attributes
     nsHttpResponseHead    *ResponseHead()   { return mHaveAllHeaders ? mResponseHead : nullptr; }
     nsISupports           *SecurityInfo()   { return mSecurityInfo; }
 
     nsIEventTarget        *ConsumerTarget() { return mConsumerTarget; }
     nsISupports           *HttpChannel()    { return mChannel; }
@@ -161,16 +165,21 @@ public:
     mozilla::TimeStamp GetRequestStart();
     mozilla::TimeStamp GetResponseStart();
     mozilla::TimeStamp GetResponseEnd();
 
     int64_t GetTransferSize() { return mTransferSize; }
 
     MOZ_MUST_USE bool Do0RTT() override;
     MOZ_MUST_USE nsresult Finish0RTT(bool aRestart, bool aAlpnChanged /* ignored */) override;
+
+    uint64_t TopLevelOuterContentWindowId()
+    {
+        return mTopLevelOuterContentWindowId;
+    }
 private:
     friend class DeleteHttpTransaction;
     virtual ~nsHttpTransaction();
 
     MOZ_MUST_USE nsresult Restart();
     char    *LocateHttpStart(char *buf, uint32_t len,
                              bool aAllowPartialMatch);
     MOZ_MUST_USE nsresult ParseLine(nsACString &line);
@@ -328,16 +337,18 @@ private:
 
     // protected by nsHttp::GetLock()
     nsHttpResponseHead             *mForTakeResponseHead;
     bool                            mResponseHeadTaken;
 
     // The time when the transaction was submitted to the Connection Manager
     TimeStamp                       mPendingTime;
 
+    uint64_t                        mTopLevelOuterContentWindowId;
+
 // For Rate Pacing via an EventTokenBucket
 public:
     // called by the connection manager to run this transaction through the
     // token bucket. If the token bucket admits the transaction immediately it
     // returns true. The function is called repeatedly until it returns true.
     bool TryToRunPacedRequest();
 
     // ATokenBucketEvent pure virtual implementation. Called by the token bucket
--- a/netwerk/protocol/websocket/PWebSocket.ipdl
+++ b/netwerk/protocol/websocket/PWebSocket.ipdl
@@ -3,21 +3,23 @@
 
 /* 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 protocol PNecko;
 include protocol PBrowser;
 include protocol PTransportProvider;
-include InputStreamParams;
+include IPCStream;
 include URIParams;
 include NeckoChannelParams;
 
 include protocol PBlob; //FIXME: bug #792908
+include protocol PFileDescriptorSet; //FIXME: bug #792908
+include protocol PSendStream; //FIXME: bug #792908
 
 using class IPC::SerializedLoadContext from "SerializedLoadContext.h";
 using struct mozilla::void_t from "ipc/IPCMessageUtils.h";
 
 namespace mozilla {
 namespace net {
 
 union OptionalTransportProvider
@@ -43,17 +45,17 @@ parent:
                   uint32_t aPingTimeout,
                   bool aClientSetPingTimeout,
                   OptionalLoadInfoArgs aLoadInfoArgs,
                   OptionalTransportProvider aProvider,
                   nsCString aNegotiatedExtensions);
   async Close(uint16_t code, nsCString reason);
   async SendMsg(nsCString aMsg);
   async SendBinaryMsg(nsCString aMsg);
-  async SendBinaryStream(InputStreamParams aStream, uint32_t aLength);
+  async SendBinaryStream(IPCStream aStream, uint32_t aLength);
 
   async DeleteSelf();
 
 child:
   // Forwarded notifications corresponding to the nsIWebSocketListener interface
   async OnStart(nsCString aProtocol, nsCString aExtensions,
                 nsString aEffectiveURL, bool aEncrypted);
   async OnStop(nsresult aStatusCode);
--- a/netwerk/protocol/websocket/WebSocketChannelChild.cpp
+++ b/netwerk/protocol/websocket/WebSocketChannelChild.cpp
@@ -6,17 +6,17 @@
 
 #include "WebSocketLog.h"
 #include "base/compiler_specific.h"
 #include "mozilla/dom/TabChild.h"
 #include "mozilla/net/NeckoChild.h"
 #include "WebSocketChannelChild.h"
 #include "nsITabChild.h"
 #include "nsNetUtil.h"
-#include "mozilla/ipc/InputStreamUtils.h"
+#include "mozilla/ipc/IPCStreamUtils.h"
 #include "mozilla/ipc/URIUtils.h"
 #include "mozilla/ipc/BackgroundUtils.h"
 #include "mozilla/net/ChannelEventQueue.h"
 #include "SerializedLoadContext.h"
 
 using namespace mozilla::ipc;
 
 namespace mozilla {
@@ -655,17 +655,17 @@ WebSocketChannelChild::SendBinaryMsg(con
 
   return NS_OK;
 }
 
 class BinaryStreamEvent : public Runnable
 {
 public:
   BinaryStreamEvent(WebSocketChannelChild *aChild,
-                    OptionalInputStreamParams *aStream,
+                    nsIInputStream* aStream,
                     uint32_t aLength)
     : mChild(aChild)
     , mStream(aStream)
     , mLength(aLength)
   {
     MOZ_RELEASE_ASSERT(!NS_IsMainThread());
     MOZ_ASSERT(aChild);
   }
@@ -675,54 +675,44 @@ public:
     nsresult rv = mChild->SendBinaryStream(mStream, mLength);
     if (NS_FAILED(rv)) {
       LOG(("WebSocketChannelChild::BinaryStreamEvent %p "
            "SendBinaryStream failed (%08" PRIx32 ")\n", this, static_cast<uint32_t>(rv)));
     }
     return NS_OK;
   }
 private:
-  RefPtr<WebSocketChannelChild>      mChild;
-  nsAutoPtr<OptionalInputStreamParams> mStream;
-  uint32_t                             mLength;
+  RefPtr<WebSocketChannelChild> mChild;
+  nsCOMPtr<nsIInputStream> mStream;
+  uint32_t mLength;
 };
 
 NS_IMETHODIMP
 WebSocketChannelChild::SendBinaryStream(nsIInputStream *aStream,
                                         uint32_t aLength)
 {
-  OptionalInputStreamParams *stream = new OptionalInputStreamParams();
-  nsTArray<mozilla::ipc::FileDescriptor> fds;
-  SerializeInputStream(aStream, *stream, fds);
-
-  MOZ_ASSERT(fds.IsEmpty());
-
   if (!NS_IsMainThread()) {
     MOZ_RELEASE_ASSERT(NS_GetCurrentThread() == mTargetThread);
-    return NS_DispatchToMainThread(new BinaryStreamEvent(this, stream, aLength));
+    return NS_DispatchToMainThread(new BinaryStreamEvent(this, aStream, aLength));
   }
-  return SendBinaryStream(stream, aLength);
-}
 
-nsresult
-WebSocketChannelChild::SendBinaryStream(OptionalInputStreamParams *aStream,
-                                        uint32_t aLength)
-{
   LOG(("WebSocketChannelChild::SendBinaryStream() %p\n", this));
 
-  nsAutoPtr<OptionalInputStreamParams> stream(aStream);
+  AutoIPCStream autoStream;
+  autoStream.Serialize(aStream,
+                       static_cast<mozilla::dom::ContentChild*>(gNeckoChild->Manager()));
 
   {
     MutexAutoLock lock(mMutex);
     if (mIPCState != Opened) {
       return NS_ERROR_UNEXPECTED;
     }
   }
 
-  if (!SendSendBinaryStream(*stream, aLength)) {
+  if (!SendSendBinaryStream(autoStream.TakeValue(), aLength)) {
     return NS_ERROR_UNEXPECTED;
   }
 
   return NS_OK;
 }
 
 NS_IMETHODIMP
 WebSocketChannelChild::GetSecurityInfo(nsISupports **aSecurityInfo)
--- a/netwerk/protocol/websocket/WebSocketChannelChild.h
+++ b/netwerk/protocol/websocket/WebSocketChannelChild.h
@@ -7,16 +7,17 @@
 #ifndef mozilla_net_WebSocketChannelChild_h
 #define mozilla_net_WebSocketChannelChild_h
 
 #include "mozilla/net/PWebSocketChild.h"
 #include "mozilla/net/BaseWebSocketChannel.h"
 #include "nsString.h"
 
 namespace mozilla {
+
 namespace net {
 
 class ChannelEvent;
 class ChannelEventQueue;
 
 class WebSocketChannelChild final : public BaseWebSocketChannel,
                                     public PWebSocketChild
 {
@@ -31,17 +32,16 @@ class WebSocketChannelChild final : publ
   NS_IMETHOD AsyncOpen(nsIURI *aURI, const nsACString &aOrigin,
                        uint64_t aInnerWindowID,
                        nsIWebSocketListener *aListener,
                        nsISupports *aContext) override;
   NS_IMETHOD Close(uint16_t code, const nsACString & reason) override;
   NS_IMETHOD SendMsg(const nsACString &aMsg) override;
   NS_IMETHOD SendBinaryMsg(const nsACString &aMsg) override;
   NS_IMETHOD SendBinaryStream(nsIInputStream *aStream, uint32_t aLength) override;
-  MOZ_MUST_USE nsresult SendBinaryStream(OptionalInputStreamParams *aStream, uint32_t aLength);
   NS_IMETHOD GetSecurityInfo(nsISupports **aSecurityInfo) override;
 
   void AddIPDLReference();
   void ReleaseIPDLReference();
 
   // Off main thread URI access.
   void GetEffectiveURL(nsAString& aEffectiveURL) const override;
   bool IsEncrypted() const override;
--- a/netwerk/protocol/websocket/WebSocketChannelParent.cpp
+++ b/netwerk/protocol/websocket/WebSocketChannelParent.cpp
@@ -182,23 +182,22 @@ WebSocketChannelParent::RecvSendBinaryMs
   if (mChannel) {
     nsresult rv = mChannel->SendBinaryMsg(aMsg);
     NS_ENSURE_SUCCESS(rv, IPC_OK());
   }
   return IPC_OK();
 }
 
 mozilla::ipc::IPCResult
-WebSocketChannelParent::RecvSendBinaryStream(const InputStreamParams& aStream,
+WebSocketChannelParent::RecvSendBinaryStream(const IPCStream& aStream,
                                              const uint32_t& aLength)
 {
   LOG(("WebSocketChannelParent::RecvSendBinaryStream() %p\n", this));
   if (mChannel) {
-    nsTArray<mozilla::ipc::FileDescriptor> fds;
-    nsCOMPtr<nsIInputStream> stream = DeserializeInputStream(aStream, fds);
+    nsCOMPtr<nsIInputStream> stream = DeserializeIPCStream(aStream);
     if (!stream) {
       return IPC_FAIL_NO_REASON(this);
     }
     nsresult rv = mChannel->SendBinaryStream(stream, aLength);
     NS_ENSURE_SUCCESS(rv, IPC_OK());
   }
   return IPC_OK();
 }
--- a/netwerk/protocol/websocket/WebSocketChannelParent.h
+++ b/netwerk/protocol/websocket/WebSocketChannelParent.h
@@ -47,17 +47,17 @@ class WebSocketChannelParent : public PW
                                         const uint32_t& aPingTimeout,
                                         const bool& aClientSetPingTimeout,
                                         const OptionalLoadInfoArgs& aLoadInfoArgs,
                                         const OptionalTransportProvider& aTransportProvider,
                                         const nsCString& aNegotiatedExtensions) override;
   mozilla::ipc::IPCResult RecvClose(const uint16_t & code, const nsCString & reason) override;
   mozilla::ipc::IPCResult RecvSendMsg(const nsCString& aMsg) override;
   mozilla::ipc::IPCResult RecvSendBinaryMsg(const nsCString& aMsg) override;
-  mozilla::ipc::IPCResult RecvSendBinaryStream(const InputStreamParams& aStream,
+  mozilla::ipc::IPCResult RecvSendBinaryStream(const IPCStream& aStream,
                                                const uint32_t& aLength) override;
   mozilla::ipc::IPCResult RecvDeleteSelf() override;
 
   void ActorDestroy(ActorDestroyReason why) override;
 
   nsCOMPtr<nsIAuthPromptProvider> mAuthProvider;
   nsCOMPtr<nsIWebSocketChannel> mChannel;
   nsCOMPtr<nsILoadContext> mLoadContext;
--- a/security/apps/moz.build
+++ b/security/apps/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files("**"):
+    BUG_COMPONENT = ("Core", "Security: PSM")
+
 UNIFIED_SOURCES += [
     'AppSignatureVerification.cpp',
     'AppTrustDomain.cpp',
 ]
 
 FINAL_LIBRARY = 'xul'
 
 LOCAL_INCLUDES += [
--- a/security/certverifier/moz.build
+++ b/security/certverifier/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files("**"):
+    BUG_COMPONENT = ("Core", "Security: PSM")
+
 EXPORTS += [
     'BRNameMatchingPolicy.h',
     'CertVerifier.h',
     'CTLog.h',
     'CTPolicyEnforcer.h',
     'CTVerifyResult.h',
     'OCSPCache.h',
     'SignedCertificateTimestamp.h',
--- a/security/manager/moz.build
+++ b/security/manager/moz.build
@@ -1,10 +1,13 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files("**"):
+    BUG_COMPONENT = ("Core", "Security: PSM")
+
 DIRS += ['ssl', 'locales']
 
 if CONFIG['MOZ_XUL'] and CONFIG['MOZ_BUILD_APP'] != 'mobile/android':
     DIRS += ['pki']
--- a/security/moz.build
+++ b/security/moz.build
@@ -1,14 +1,26 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files("**"):
+    BUG_COMPONENT = ("Core", "Security: PSM")
+
+with Files("generate*.py"):
+    BUG_COMPONENT = ("Core", "Build Config")
+
+with Files("nss/**"):
+    BUG_COMPONENT = ("NSS", "Libraries")
+
+with Files("nss.symbols"):
+    BUG_COMPONENT = ("NSS", "Libraries")
+
 if CONFIG['MOZ_SYSTEM_NSS']:
     Library('nss')
     OS_LIBS += CONFIG['NSS_LIBS']
 else:
     include('/build/gyp_base.mozbuild')
     if CONFIG['MOZ_FOLD_LIBS']:
         GeckoSharedLibrary('nss', linkage=None)
         # TODO: The library name can be changed when bug 845217 is fixed.
deleted file mode 100644
--- a/security/patches/README
+++ /dev/null
@@ -1,2 +0,0 @@
-This directory contains patches that were added locally
-on top of the NSS release.
--- a/security/pkix/moz.build
+++ b/security/pkix/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files("**"):
+    BUG_COMPONENT = ("Core", "Security: PSM")
+
 SOURCES += [
     'lib/pkixbuild.cpp',
     'lib/pkixcert.cpp',
     'lib/pkixcheck.cpp',
     'lib/pkixder.cpp',
     'lib/pkixnames.cpp',
     'lib/pkixnss.cpp',
     'lib/pkixocsp.cpp',
--- a/services/blocklists/moz.build
+++ b/services/blocklists/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; c-basic-offset: 4; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Firefox', 'Sync')
+
 FINAL_TARGET_FILES.defaults.blocklists += ['addons.json',
                                            'certificates.json',
                                            'gfx.json',
                                            'plugins.json']
 
 FINAL_TARGET_FILES.defaults.pinning += ['pins.json']
 
 if CONFIG['MOZ_BUILD_APP'] == 'browser':
--- a/services/cloudsync/moz.build
+++ b/services/cloudsync/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Cloud Services', 'cloudSync')
+
 SPHINX_TREES['cloudsync'] = 'docs'
 
 EXTRA_JS_MODULES += [
   'CloudSync.jsm',
   'CloudSyncAdapters.jsm',
   'CloudSyncBookmarks.jsm',
   'CloudSyncBookmarksFolderCache.jsm',
   'CloudSyncEventSource.jsm',
--- a/services/common/moz.build
+++ b/services/common/moz.build
@@ -1,16 +1,16 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
 with Files('**'):
-    BUG_COMPONENT = ('Mozilla Services', 'Firefox: Common')
+    BUG_COMPONENT = ('Cloud Services', 'Firefox: Common')
 
 TEST_DIRS += ['tests']
 
 EXTRA_COMPONENTS += [
     'servicesComponents.manifest',
 ]
 
 EXTRA_JS_MODULES['services-common'] += [
--- a/services/crypto/moz.build
+++ b/services/crypto/moz.build
@@ -1,16 +1,16 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
 with Files('**'):
-    BUG_COMPONENT = ('Mozilla Services', 'Firefox Sync: Crypto')
+    BUG_COMPONENT = ('Firefox', 'Sync')
 
 DIRS += ['component']
 
 XPCSHELL_TESTS_MANIFESTS += ['tests/unit/xpcshell.ini']
 
 EXTRA_JS_MODULES['services-crypto'] += [
     'modules/jwcrypto.jsm',
     'modules/utils.js',
--- a/services/fxaccounts/moz.build
+++ b/services/fxaccounts/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Core', 'fxAccounts')
+
 DIRS += ['interfaces']
 
 MOCHITEST_CHROME_MANIFESTS += ['tests/mochitest/chrome.ini']
 
 BROWSER_CHROME_MANIFESTS += ['tests/browser/browser.ini']
 
 XPCSHELL_TESTS_MANIFESTS += ['tests/xpcshell/xpcshell.ini']
 
--- a/services/moz.build
+++ b/services/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('moz.build'):
+    BUG_COMPONENT = ('Core', 'Build Config')
+
 DIRS += [
     'common',
     'crypto',
 ]
 
 if CONFIG['MOZ_WIDGET_TOOLKIT'] != 'android':
     DIRS += [
         'fxaccounts',
--- a/services/sync/moz.build
+++ b/services/sync/moz.build
@@ -1,16 +1,16 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
 with Files('**'):
-    BUG_COMPONENT = ('Mozilla Services', 'Firefox Sync: Backend')
+    BUG_COMPONENT = ('Firefox', 'Sync')
 
 DIRS += ['locales']
 
 XPCSHELL_TESTS_MANIFESTS += ['tests/unit/xpcshell.ini']
 
 EXTRA_COMPONENTS += [
     'SyncComponents.manifest',
     'Weave.js',
--- a/testing/mozharness/scripts/release/uptake_monitoring.py
+++ b/testing/mozharness/scripts/release/uptake_monitoring.py
@@ -72,18 +72,19 @@ class UptakeMonitoring(BaseScript, Virtu
         props = self.buildbot_config["properties"]
         for prop in ['tuxedo_server_url', 'version']:
             if props.get(prop):
                 self.info("Overriding %s with %s" % (prop, props[prop]))
                 self.config[prop] = props.get(prop)
             else:
                 self.warning("%s could not be found within buildprops" % prop)
                 return
-        partials = [v.strip() for v in props["partial_versions"].split(",")]
-        self.config["partial_versions"] = [v.split("build")[0] for v in partials]
+        if props.get('partial_versions'):
+            partials = [v.strip() for v in props["partial_versions"].split(",")]
+            self.config["partial_versions"] = [v.split("build")[0] for v in partials]
         self.config["platforms"] = [p.strip() for p in
                                     props["platforms"].split(",")]
 
     def _get_product_uptake(self, tuxedo_server_url, auth,
                             related_product, os):
         from redo import retry
         import requests
 
@@ -131,18 +132,18 @@ class UptakeMonitoring(BaseScript, Virtu
                 paths_platforms = set(info["paths"].keys())
                 platforms = enUS_platforms.intersection(paths_platforms)
 
                 for platform in platforms:
                     bouncer_platform = info["paths"].get(platform).get('bouncer-platform')
                     dl.append(self._get_product_uptake(tuxedo_server_url, auth,
                                                        related_product, bouncer_platform))
         # handle the partials as well
-        prev_versions = self.config["partial_versions"]
-        for product, info in self.config["partials"].iteritems():
+        prev_versions = self.config.get("partial_versions", [])
+        for product, info in self.config.get("partials", {}).iteritems():
             if info.get("check_uptake"):
                 product_template = info["product-name"]
                 for prev_version in prev_versions:
                     subs = {
                         "version": version,
                         "prev_version": prev_version
                     }
                     related_product = product_template % subs
--- a/toolkit/components/passwordmgr/test/browser/browser.ini
+++ b/toolkit/components/passwordmgr/test/browser/browser.ini
@@ -35,17 +35,17 @@ support-files =
   subtst_notifications_1.html
   subtst_notifications_8.html
 [browser_capture_doorhanger_window_open.js]
 support-files =
   subtst_notifications_11.html
   subtst_notifications_11_popup.html
 skip-if = os == "linux" # Bug 1312981, bug 1313136
 [browser_context_menu_autocomplete_interaction.js]
-skip-if = asan || (os == 'linux' && debug && (bits == 32)) # disabled on asan and linux 32bit debug because of performance-related intermittents (see bug 1337772)
+skip-if = asan || (os == 'linux') # disabled on asan and linux * (see bug 1337772)
 [browser_username_select_dialog.js]
 support-files =
   subtst_notifications_change_p.html
 [browser_DOMFormHasPassword.js]
 [browser_DOMInputPasswordAdded.js]
 [browser_exceptions_dialog.js]
 [browser_formless_submit_chrome.js]
 [browser_hasInsecureLoginForms.js]
--- a/toolkit/components/telemetry/.flake8
+++ b/toolkit/components/telemetry/.flake8
@@ -1,5 +1,5 @@
 [flake8]
 # See http://pep8.readthedocs.io/en/latest/intro.html#configuration
-ignore = E121, E123, E126, E129, E133, E226, E241, E242, E704, W503, E402, E302, E703, E502, E128, E501, E713, E231, E111, E261, E222, E203, E201, E202, W602, E127, F841, F401, W601
+ignore = E121, E123, E126, E129, E133, E226, E241, E242, E704, W503, E402, E703, E502, E128, E501, E713, E231, E111, E261, E222, E203, E201, E202, W602, E127, F841, F401, W601
 max-line-length = 99
 filename = *.py, +.lint
--- a/toolkit/components/telemetry/TelemetryEnvironment.jsm
+++ b/toolkit/components/telemetry/TelemetryEnvironment.jsm
@@ -7,17 +7,16 @@
 this.EXPORTED_SYMBOLS = [
   "TelemetryEnvironment",
 ];
 
 const {classes: Cc, interfaces: Ci, utils: Cu} = Components;
 const myScope = this;
 
 Cu.import("resource://gre/modules/XPCOMUtils.jsm", this);
-Cu.import("resource://gre/modules/Task.jsm");
 Cu.import("resource://gre/modules/Log.jsm");
 Cu.import("resource://gre/modules/Preferences.jsm");
 Cu.import("resource://gre/modules/PromiseUtils.jsm");
 Cu.import("resource://gre/modules/Services.jsm");
 Cu.import("resource://gre/modules/TelemetryUtils.jsm", this);
 Cu.import("resource://gre/modules/ObjectUtils.jsm");
 Cu.import("resource://gre/modules/TelemetryController.jsm", this);
 Cu.import("resource://gre/modules/AppConstants.jsm");
@@ -513,56 +512,56 @@ EnvironmentAddonBuilder.prototype = {
    *
    * This should only be called from _pendingTask; otherwise we risk
    * running this during addon manager shutdown.
    *
    * @returns Promise<Object> This returns a Promise resolved with a status object with the following members:
    *   changed - Whether the environment changed.
    *   oldEnvironment - Only set if a change occured, contains the environment data before the change.
    */
-  _updateAddons: Task.async(function* () {
+  async _updateAddons() {
     this._environment._log.trace("_updateAddons");
     let personaId = null;
     if (AppConstants.platform !== "gonk") {
       let theme = LightweightThemeManager.currentTheme;
       if (theme) {
         personaId = theme.id;
       }
     }
 
     let addons = {
-      activeAddons: yield this._getActiveAddons(),
-      theme: yield this._getActiveTheme(),
+      activeAddons: await this._getActiveAddons(),
+      theme: await this._getActiveTheme(),
       activePlugins: this._getActivePlugins(),
-      activeGMPlugins: yield this._getActiveGMPlugins(),
+      activeGMPlugins: await this._getActiveGMPlugins(),
       activeExperiment: this._getActiveExperiment(),
       persona: personaId,
     };
 
     let result = {
       changed: !this._environment._currentEnvironment.addons ||
                !ObjectUtils.deepEqual(addons, this._environment._currentEnvironment.addons),
     };
 
     if (result.changed) {
       this._environment._log.trace("_updateAddons: addons differ");
       result.oldEnvironment = Cu.cloneInto(this._environment._currentEnvironment, myScope);
       this._environment._currentEnvironment.addons = addons;
     }
 
     return result;
-  }),
+  },
 
   /**
    * Get the addon data in object form.
    * @return Promise<object> containing the addon data.
    */
-  _getActiveAddons: Task.async(function* () {
+  async _getActiveAddons() {
     // Request addons, asynchronously.
-    let allAddons = yield AddonManager.getAddonsByTypes(["extension", "service"]);
+    let allAddons = await AddonManager.getAddonsByTypes(["extension", "service"]);
 
     let activeAddons = {};
     for (let addon of allAddons) {
       // Skip addons which are not active.
       if (!addon.isActive) {
         continue;
       }
 
@@ -595,25 +594,25 @@ EnvironmentAddonBuilder.prototype = {
 
       } catch (ex) {
         this._environment._log.error("_getActiveAddons - An addon was discarded due to an error", ex);
         continue;
       }
     }
 
     return activeAddons;
-  }),
+  },
 
   /**
    * Get the currently active theme data in object form.
    * @return Promise<object> containing the active theme data.
    */
-  _getActiveTheme: Task.async(function* () {
+  async _getActiveTheme() {
     // Request themes, asynchronously.
-    let themes = yield AddonManager.getAddonsByTypes(["theme"]);
+    let themes = await AddonManager.getAddonsByTypes(["theme"]);
 
     let activeTheme = {};
     // We only store information about the active theme.
     let theme = themes.find(theme => theme.isActive);
     if (theme) {
       // Make sure to have valid dates.
       let installDate = new Date(Math.max(0, theme.installDate));
       let updateDate = new Date(Math.max(0, theme.updateDate));
@@ -630,17 +629,17 @@ EnvironmentAddonBuilder.prototype = {
         foreignInstall: enforceBoolean(theme.foreignInstall),
         hasBinaryComponents: theme.hasBinaryComponents,
         installDay: Utils.millisecondsToDays(installDate.getTime()),
         updateDay: Utils.millisecondsToDays(updateDate.getTime()),
       };
     }
 
     return activeTheme;
-  }),
+  },
 
   /**
    * Get the plugins data in object form.
    * @return Object containing the plugins data.
    */
   _getActivePlugins() {
     let pluginTags =
       Cc["@mozilla.org/plugin/host;1"].getService(Ci.nsIPluginHost).getPluginTags({});
@@ -677,19 +676,19 @@ EnvironmentAddonBuilder.prototype = {
 
   /**
    * Get the GMPlugins data in object form.
    * @return Object containing the GMPlugins data.
    *
    * This should only be called from _pendingTask; otherwise we risk
    * running this during addon manager shutdown.
    */
-  _getActiveGMPlugins: Task.async(function* () {
+  async _getActiveGMPlugins() {
     // Request plugins, asynchronously.
-    let allPlugins = yield AddonManager.getAddonsByTypes(["plugin"]);
+    let allPlugins = await AddonManager.getAddonsByTypes(["plugin"]);
 
     let activeGMPlugins = {};
     for (let plugin of allPlugins) {
       // Only get info for active GMplugins.
       if (!plugin.isGMPlugin || !plugin.isActive) {
         continue;
       }
 
@@ -701,17 +700,17 @@ EnvironmentAddonBuilder.prototype = {
         };
       } catch (ex) {
         this._environment._log.error("_getActiveGMPlugins - A GMPlugin was discarded due to an error", ex);
         continue;
       }
     }
 
     return activeGMPlugins;
-  }),
+  },
 
   /**
    * Get the active experiment data in object form.
    * @return Object containing the active experiment data.
    */
   _getActiveExperiment() {
     let experimentInfo = {};
     try {
@@ -1193,45 +1192,45 @@ EnvironmentCache.prototype = {
 
     this._updateSearchEngine();
   },
 
   /**
    * Update the cached profile data.
    * @returns Promise<> resolved when the I/O is complete.
    */
-  _updateProfile: Task.async(function* () {
+  async _updateProfile() {
     const logger = Log.repository.getLoggerWithMessagePrefix(LOGGER_NAME, "ProfileAge - ");
     let profileAccessor = new ProfileAge(null, logger);
 
-    let creationDate = yield profileAccessor.created;
-    let resetDate = yield profileAccessor.reset;
+    let creationDate = await profileAccessor.created;
+    let resetDate = await profileAccessor.reset;
 
     this._currentEnvironment.profile.creationDate =
       Utils.millisecondsToDays(creationDate);
     if (resetDate) {
       this._currentEnvironment.profile.resetDate =
         Utils.millisecondsToDays(resetDate);
     }
-  }),
+  },
 
   /**
    * Update the cached attribution data object.
    * @returns Promise<> resolved when the I/O is complete.
    */
-  _updateAttribution: Task.async(function* () {
-    let data = yield AttributionCode.getAttrDataAsync();
+  async _updateAttribution() {
+    let data = await AttributionCode.getAttrDataAsync();
     if (Object.keys(data).length > 0) {
       this._currentEnvironment.settings.attribution = {};
       for (let key in data) {
         this._currentEnvironment.settings.attribution[key] =
           limitStringToLength(data[key], MAX_ATTRIBUTION_STRING_LENGTH);
       }
     }
-  }),
+  },
 
   /**
    * Get the partner data in object form.
    * @return Object containing the partner data.
    */
   _getPartner() {
     let partnerData = {
       distributionId: Preferences.get(PREF_DISTRIBUTION_ID, null),
--- a/toolkit/components/telemetry/gen-event-data.py
+++ b/toolkit/components/telemetry/gen-event-data.py
@@ -24,16 +24,17 @@ file_header = """\
 namespace {
 """
 
 file_footer = """\
 } // namespace
 #endif // mozilla_TelemetryEventData_h
 """
 
+
 def write_extra_table(events, output, string_table):
     table_name = "gExtraKeysTable"
     extra_table = []
     extra_count = 0
 
     print("const uint32_t %s[] = {" % table_name, file=output)
 
     for e in events:
@@ -56,16 +57,17 @@ def write_extra_table(events, output, st
         extra_table.append((extra_index, len(extra_keys)))
 
     print("};", file=output)
     static_assert(output, "sizeof(%s) <= UINT32_MAX" % table_name,
                   "index overflow")
 
     return extra_table
 
+
 def write_common_event_table(events, output, string_table, extra_table):
     table_name = "gCommonEventInfo"
     extra_count = 0
 
     print("const CommonEventInfo %s[] = {" % table_name, file=output)
     for e,extras in zip(events, extra_table):
         # Write a comment to make the file human-readable.
         print("  // category: %s" % e.category, file=output)
@@ -82,16 +84,17 @@ def write_common_event_table(events, out
                  e.dataset,
                  " | ".join(e.record_in_processes_enum)),
               file=output)
 
     print("};", file=output)
     static_assert(output, "sizeof(%s) <= UINT32_MAX" % table_name,
                   "index overflow")
 
+
 def write_event_table(events, output, string_table):
     table_name = "gEventInfo"
     print("const EventInfo %s[] = {" % table_name, file=output)
 
     for common_info_index,e in enumerate(events):
         for method_name, object_name in itertools.product(e.methods, e.objects):
             print("  // category: %s, method: %s, object: %s" %
                     (e.category, method_name, object_name),
@@ -102,16 +105,17 @@ def write_event_table(events, output, st
                      string_table.stringIndex(method_name),
                      string_table.stringIndex(object_name)),
                   file=output)
 
     print("};", file=output)
     static_assert(output, "sizeof(%s) <= UINT32_MAX" % table_name,
                   "index overflow")
 
+
 def main(output, *filenames):
     # Load the event data.
     if len(filenames) > 1:
         raise Exception('We don\'t support loading from more than one file.')
     events = parse_events.load_events(filenames[0])
 
     # Write the scalar data file.
     print(banner, file=output)
--- a/toolkit/components/telemetry/gen-event-enum.py
+++ b/toolkit/components/telemetry/gen-event-enum.py
@@ -24,16 +24,17 @@ namespace EventID {
 
 file_footer = """\
 } // namespace EventID
 } // namespace mozilla
 } // namespace Telemetry
 #endif // mozilla_TelemetryEventEnums_h
 """
 
+
 def main(output, *filenames):
     # Load the events first.
     if len(filenames) > 1:
         raise Exception('We don\'t support loading from more than one file.')
     events = parse_events.load_events(filenames[0])
 
     grouped = dict()
     index = 0
--- a/toolkit/components/telemetry/gen-histogram-bucket-ranges.py
+++ b/toolkit/components/telemetry/gen-histogram-bucket-ranges.py
@@ -8,16 +8,17 @@
 
 import sys
 import re
 import histogram_tools
 import json
 
 from collections import OrderedDict
 
+
 def main(argv):
     filenames = argv
 
     all_histograms = OrderedDict()
 
     for histogram in histogram_tools.from_files(filenames):
         name = histogram.name()
         parameters = OrderedDict()
--- a/toolkit/components/telemetry/gen-histogram-data.py
+++ b/toolkit/components/telemetry/gen-histogram-data.py
@@ -10,16 +10,17 @@ from shared_telemetry_utils import Strin
 
 import sys
 import histogram_tools
 import itertools
 
 banner = """/* This file is auto-generated, see gen-histogram-data.py.  */
 """
 
+
 def print_array_entry(output, histogram, name_index, exp_index, label_index, label_count):
     cpp_guard = histogram.cpp_guard()
     if cpp_guard:
         print("#if defined(%s)" % cpp_guard, file=output)
     print("  { %s, %s, %s, %s, %d, %d, %s, %d, %d, %s }," \
         % (histogram.low(),
            histogram.high(),
            histogram.n_buckets(),
@@ -28,16 +29,17 @@ def print_array_entry(output, histogram,
            exp_index,
            histogram.dataset(),
            label_index,
            label_count,
            "true" if histogram.keyed() else "false"), file=output)
     if cpp_guard:
         print("#endif", file=output)
 
+
 def write_histogram_table(output, histograms):
     string_table = StringTable()
     label_table = []
     label_count = 0
 
     print("const HistogramInfo gHistograms[] = {", file=output)
     for histogram in histograms:
         name_index = string_table.stringIndex(histogram.name())
@@ -69,44 +71,51 @@ def write_histogram_table(output, histog
 # Write out static asserts for histogram data.  We'd prefer to perform
 # these checks in this script itself, but since several histograms
 # (generally enumerated histograms) use compile-time constants for
 # their upper bounds, we have to let the compiler do the checking.
 
 def static_asserts_for_boolean(output, histogram):
     pass
 
+
 def static_asserts_for_flag(output, histogram):
     pass
 
+
 def static_asserts_for_count(output, histogram):
     pass
 
+
 def static_asserts_for_enumerated(output, histogram):
     n_values = histogram.high()
     static_assert(output, "%s > 2" % n_values,
                   "Not enough values for %s" % histogram.name())
 
+
 def shared_static_asserts(output, histogram):
     name = histogram.name()
     low = histogram.low()
     high = histogram.high()
     n_buckets = histogram.n_buckets()
     static_assert(output, "%s < %s" % (low, high), "low >= high for %s" % name)
     static_assert(output, "%s > 2" % n_buckets, "Not enough values for %s" % name)
     static_assert(output, "%s >= 1" % low, "Incorrect low value for %s" % name)
     static_assert(output, "%s > %s" % (high, n_buckets),
                   "high must be > number of buckets for %s; you may want an enumerated histogram" % name)
 
+
 def static_asserts_for_linear(output, histogram):
     shared_static_asserts(output, histogram)
 
+
 def static_asserts_for_exponential(output, histogram):
     shared_static_asserts(output, histogram)
 
+
 def write_histogram_static_asserts(output, histograms):
     print("""
 // Perform the checks at the beginning of HistogramGet at
 // compile time, so that incorrect histogram definitions
 // give compile-time errors, not runtime errors.""", file=output)
 
     table = {
         'boolean' : static_asserts_for_boolean,
@@ -117,16 +126,17 @@ def write_histogram_static_asserts(outpu
         'linear' : static_asserts_for_linear,
         'exponential' : static_asserts_for_exponential,
         }
 
     for histogram in histograms:
         histogram_tools.table_dispatch(histogram.kind(), table,
                                        lambda f: f(output, histogram))
 
+
 def write_debug_histogram_ranges(output, histograms):
     ranges_lengths = []
 
     # Collect all the range information from individual histograms.
     # Write that information out as well.
     print("#ifdef DEBUG", file=output)
     print("const int gBucketLowerBounds[] = {", file=output)
     for histogram in histograms:
@@ -161,16 +171,17 @@ def write_debug_histogram_ranges(output,
             print("#if defined(%s)" % cpp_guard, file=output)
         print("{ %d, %d }," % (offset, range_length), file=output)
         if cpp_guard:
             print("#endif", file=output)
         offset += range_length
     print("};", file=output)
     print("#endif", file=output)
 
+
 def main(output, *filenames):
     histograms = list(histogram_tools.from_files(filenames))
 
     print(banner, file=output)
     write_histogram_table(output, histograms)
     write_histogram_static_asserts(output, histograms)
     write_debug_histogram_ranges(output, histograms)
 
--- a/toolkit/components/telemetry/gen-histogram-enum.py
+++ b/toolkit/components/telemetry/gen-histogram-enum.py
@@ -31,16 +31,17 @@ namespace mozilla {
 namespace Telemetry {
 """
 
 footer = """
 } // namespace mozilla
 } // namespace Telemetry
 #endif // mozilla_TelemetryHistogramEnums_h"""
 
+
 def main(output, *filenames):
     # Print header.
     print(banner, file=output)
     print(header, file=output)
 
     # Load the histograms.
     all_histograms = list(histogram_tools.from_files(filenames))
     groups = itertools.groupby(all_histograms,
--- a/toolkit/components/telemetry/gen-scalar-data.py
+++ b/toolkit/components/telemetry/gen-scalar-data.py
@@ -23,16 +23,17 @@ file_header = """\
 namespace {
 """
 
 file_footer = """\
 } // namespace
 #endif // mozilla_TelemetryScalarData_h
 """
 
+
 def write_scalar_info(scalar, output, name_index, expiration_index):
     """Writes a scalar entry to the output file.
 
     :param scalar: a ScalarType instance describing the scalar.
     :param output: the output stream.
     :param name_index: the index of the scalar name in the strings table.
     :param expiration_index: the index of the expiration version in the strings table.
     """
@@ -47,16 +48,17 @@ def write_scalar_info(scalar, output, na
                   scalar.dataset,
                   " | ".join(scalar.record_in_processes_enum),
                   "true" if scalar.keyed else "false"),
           file=output)
 
     if cpp_guard:
         print("#endif", file=output)
 
+
 def write_scalar_tables(scalars, output):
     """Writes the scalar and strings tables to an header file.
 
     :param scalars: a list of ScalarType instances describing the scalars.
     :param output: the output stream.
     """
     string_table = StringTable()
 
@@ -70,16 +72,17 @@ def write_scalar_tables(scalars, output)
         write_scalar_info(s, output, name_index, exp_index)
     print("};", file=output)
 
     string_table_name = "gScalarsStringTable"
     string_table.writeDefinition(output, string_table_name)
     static_assert(output, "sizeof(%s) <= UINT32_MAX" % string_table_name,
                   "index overflow")
 
+
 def main(output, *filenames):
     # Load the scalars first.
     if len(filenames) > 1:
         raise Exception('We don\'t support loading from more than one file.')
     scalars = parse_scalars.load_scalars(filenames[0])
 
     # Write the scalar data file.
     print(banner, file=output)
--- a/toolkit/components/telemetry/gen-scalar-enum.py
+++ b/toolkit/components/telemetry/gen-scalar-enum.py
@@ -25,16 +25,17 @@ enum class ScalarID : uint32_t {\
 
 file_footer = """\
 };
 } // namespace mozilla
 } // namespace Telemetry
 #endif // mozilla_TelemetryScalarEnums_h
 """
 
+
 def main(output, *filenames):
     # Load the scalars first.
     if len(filenames) > 1:
         raise Exception('We don\'t support loading from more than one file.')
     scalars = parse_scalars.load_scalars(filenames[0])
 
     # Write the enum file.
     print(banner, file=output)
--- a/toolkit/components/telemetry/histogram_tools.py
+++ b/toolkit/components/telemetry/histogram_tools.py
@@ -28,35 +28,39 @@ try:
 except ImportError:
     # Must be in an out-of-tree usage scenario.  Trust that whoever is
     # running this script knows we need the usecounters module and has
     # ensured it's in our sys.path.
     pass
 
 from collections import OrderedDict
 
+
 def table_dispatch(kind, table, body):
     """Call body with table[kind] if it exists.  Raise an error otherwise."""
     if kind in table:
         return body(table[kind])
     else:
         raise BaseException, "don't know how to handle a histogram of kind %s" % kind
 
+
 class DefinitionException(BaseException):
     pass
 
+
 def linear_buckets(dmin, dmax, n_buckets):
     ret_array = [0] * n_buckets
     dmin = float(dmin)
     dmax = float(dmax)
     for i in range(1, n_buckets):
         linear_range = (dmin * (n_buckets - 1 - i) + dmax * (i - 1)) / (n_buckets - 2)
         ret_array[i] = int(linear_range + 0.5)
     return ret_array
 
+
 def exponential_buckets(dmin, dmax, n_buckets):
     log_max = math.log(dmax);
     bucket_index = 2;
     ret_array = [0] * n_buckets
     current = dmin
     ret_array[1] = current
     for bucket_index in range(2, n_buckets):
         log_current = math.log(current)
@@ -83,16 +87,17 @@ try:
             for name, whitelist in whitelists.iteritems():
               whitelists[name] = set(whitelist)
         except ValueError, e:
             raise BaseException, 'error parsing whitelist (%s)' % whitelist_path
 except IOError:
     whitelists = None
     print 'Unable to parse whitelist (%s). Assuming all histograms are acceptable.' % whitelist_path
 
+
 class Histogram:
     """A class for representing a histogram definition."""
 
     def __init__(self, name, definition, strict_type_checks=False):
         """Initialize a histogram named name with the given definition.
 definition is a dict-like object that must contain at least the keys:
 
  - 'kind': The kind of histogram.  Must be one of 'boolean', 'flag',
@@ -417,31 +422,34 @@ associated with the histogram.  Returns 
         return (1, n_values, n_values + 1)
 
     @staticmethod
     def exponential_bucket_parameters(definition):
         return (definition.get('low', 1),
                 definition['high'],
                 definition['n_buckets'])
 
+
 # We support generating histograms from multiple different input files, not
 # just Histograms.json.  For each file's basename, we have a specific
 # routine to parse that file, and return a dictionary mapping histogram
 # names to histogram parameters.
 def from_Histograms_json(filename):
     with open(filename, 'r') as f:
         try:
             histograms = json.load(f, object_pairs_hook=OrderedDict)
         except ValueError, e:
             raise BaseException, "error parsing histograms in %s: %s" % (filename, e.message)
     return histograms
 
+
 def from_UseCounters_conf(filename):
     return usecounters.generate_histograms(filename)
 
+
 def from_nsDeprecatedOperationList(filename):
     operation_regex = re.compile('^DEPRECATED_OPERATION\\(([^)]+)\\)')
     histograms = collections.OrderedDict()
 
     with open(filename, 'r') as f:
         for line in f:
             match = operation_regex.search(line)
             if not match:
@@ -470,16 +478,17 @@ FILENAME_PARSERS = {
 # available, so handle that gracefully.
 try:
     import usecounters
 
     FILENAME_PARSERS['UseCounters.conf'] = from_UseCounters_conf
 except ImportError:
     pass
 
+
 def from_files(filenames):
     """Return an iterator that provides a sequence of Histograms for
 the histograms defined in filenames.
     """
     all_histograms = OrderedDict()
     for filename in filenames:
         parser = FILENAME_PARSERS[os.path.basename(filename)]
         histograms = parser(filename)
--- a/toolkit/components/telemetry/parse_events.py
+++ b/toolkit/components/telemetry/parse_events.py
@@ -13,31 +13,35 @@ MAX_CATEGORY_NAME_LENGTH = 30
 MAX_METHOD_NAME_LENGTH = 20
 MAX_OBJECT_NAME_LENGTH = 20
 MAX_EXTRA_KEYS_COUNT = 10
 MAX_EXTRA_KEY_NAME_LENGTH = 15
 
 IDENTIFIER_PATTERN = r'^[a-zA-Z][a-zA-Z0-9_.]+[a-zA-Z0-9]$'
 DATE_PATTERN = r'^[0-9]{4}-[0-9]{2}-[0-9]{2}$'
 
+
 def nice_type_name(t):
     if isinstance(t, basestring):
         return "string"
     return t.__name__
 
+
 def convert_to_cpp_identifier(s, sep):
     return string.capwords(s, sep).replace(sep, "")
 
+
 class OneOf:
     """This is a placeholder type for the TypeChecker below.
     It signals that the checked value should match one of the following arguments
     passed to the TypeChecker constructor.
     """
     pass
 
+
 class TypeChecker:
     """This implements a convenience type TypeChecker to make the validation code more readable."""
     def __init__(self, kind, *args):
         """This takes 1-3 arguments, specifying the value type to check for.
         It supports:
         - atomic values, e.g.: TypeChecker(int)
         - list values, e.g.: TypeChecker(list, basestring)
         - dict values, e.g.: TypeChecker(dict, basestring, int)
@@ -87,16 +91,17 @@ class TypeChecker:
             for k,v in value.iteritems():
                 if not isinstance(x, self._args[1]):
                     raise ValueError, "%s: failed dict type check for %s - expected value type %s for key %s, got %s" %\
                                       (identifier, key,
                                        nice_type_name(self._args[1]),
                                        k,
                                        nice_type_name(type(x)))
 
+
 def type_check_event_fields(identifier, name, definition):
     """Perform a type/schema check on the event definition."""
     REQUIRED_FIELDS = {
         'objects': TypeChecker(list, basestring),
         'bug_numbers': TypeChecker(list, int),
         'notification_emails': TypeChecker(list, basestring),
         'record_in_processes': TypeChecker(list, basestring),
         'description': TypeChecker(basestring),
@@ -120,29 +125,31 @@ def type_check_event_fields(identifier, 
     unknown_fields = [f for f in definition.keys() if f not in ALL_FIELDS]
     if len(unknown_fields) > 0:
         raise KeyError(identifier + ' - unknown fields: ' + ', '.join(unknown_fields))
 
     # Type-check fields.
     for k,v in definition.iteritems():
         ALL_FIELDS[k].check(identifier, k, v)
 
+
 def string_check(identifier, field, value, min_length=1, max_length=None, regex=None):
     # Length check.
     if len(value) < min_length:
         raise ValueError("%s: value '%s' for field %s is less than minimum length of %d" %\
                          (identifier, value, field, min_length))
     if max_length and len(value) > max_length:
         raise ValueError("%s: value '%s' for field %s is greater than maximum length of %d" %\
                          (identifier, value, field, max_length))
     # Regex check.
     if regex and not re.match(regex, value):
         raise ValueError, '%s: string value "%s" for %s is not matching pattern "%s"' % \
                           (identifier, value, field, regex)
 
+
 class EventData:
     """A class representing one event."""
 
     def __init__(self, category, name, definition):
         self._category = category
         self._name = name
         self._definition = definition
 
@@ -270,16 +277,17 @@ class EventData:
             return 'nsITelemetry::DATASET_RELEASE_CHANNEL_OPTOUT'
         else:
             return 'nsITelemetry::DATASET_RELEASE_CHANNEL_OPTIN'
 
     @property
     def extra_keys(self):
         return self._definition.get('extra_keys', {}).keys()
 
+
 def load_events(filename):
     """Parses a YAML file containing the event definitions.
 
     :param filename: the YAML file containing the event definitions.
     :raises Exception: if the event file cannot be opened or parsed.
     """
 
     # Parse the event definitions from the YAML file.
--- a/toolkit/components/telemetry/parse_scalars.py
+++ b/toolkit/components/telemetry/parse_scalars.py
@@ -9,16 +9,17 @@ import shared_telemetry_utils as utils
 # The map of containing the allowed scalar types and their mapping to
 # nsITelemetry::SCALAR_* type constants.
 SCALAR_TYPES_MAP = {
     'uint': 'nsITelemetry::SCALAR_COUNT',
     'string': 'nsITelemetry::SCALAR_STRING',
     'boolean': 'nsITelemetry::SCALAR_BOOLEAN'
 }
 
+
 class ScalarType:
     """A class for representing a scalar definition."""
 
     def __init__(self, group_name, probe_name, definition):
         # Validate and set the name, so we don't need to pass it to the other
         # validation functions.
         self.validate_names(group_name, probe_name)
         self._name = probe_name
@@ -238,16 +239,17 @@ class ScalarType:
         return 'nsITelemetry::' +  ('DATASET_RELEASE_CHANNEL_OPTOUT' \
             if release_channel_collection == 'opt-out' else 'DATASET_RELEASE_CHANNEL_OPTIN')
 
     @property
     def cpp_guard(self):
         """Get the cpp guard for this scalar"""
         return self._definition.get('cpp_guard')
 
+
 def load_scalars(filename):
     """Parses a YAML file containing the scalar definition.
 
     :param filename: the YAML file containing the scalars definition.
     :raises Exception: if the scalar file cannot be opened or parsed.
     """
 
     # Parse the scalar definitions from the YAML file.
--- a/toolkit/components/telemetry/shared_telemetry_utils.py
+++ b/toolkit/components/telemetry/shared_telemetry_utils.py
@@ -16,22 +16,25 @@ KNOWN_PROCESS_FLAGS = {
     'all_childs': 'AllChilds',
     'main': 'Main',
     'content': 'Content',
     'gpu': 'Gpu',
 }
 
 PROCESS_ENUM_PREFIX = "mozilla::Telemetry::Common::RecordedProcessType::"
 
+
 def is_valid_process_name(name):
     return (name in KNOWN_PROCESS_FLAGS)
 
+
 def process_name_to_enum(name):
     return PROCESS_ENUM_PREFIX + KNOWN_PROCESS_FLAGS.get(name)
 
+
 class StringTable:
     """Manages a string table and allows C style serialization to a file."""
 
     def __init__(self):
         self.current_index = 0;
         self.table = {}
 
     def c_strlen(self, string):
@@ -92,25 +95,27 @@ class StringTable:
             e = explodeToCharArray(string)
             if e:
                 f.write("  /* %5d - \"%s\" */ %s, '\\0',\n"
                         % (offset, string, explodeToCharArray(string)))
             else:
                 f.write("  /* %5d - \"%s\" */ '\\0',\n" % (offset, string))
         f.write("};\n\n")
 
+
 def static_assert(output, expression, message):
     """Writes a C++ compile-time assertion expression to a file.
     :param output: the output stream.
     :param expression: the expression to check.
     :param message: the string literal that will appear if the expression evaluates to
         false.
     """
     print("static_assert(%s, \"%s\");" % (expression, message), file=output)
 
+
 def add_expiration_postfix(expiration):
     """ Formats the expiration version and adds a version postfix if needed.
 
     :param expiration: the expiration version string.
     :return: the modified expiration string.
     """
     if re.match(r'^[1-9][0-9]*$', expiration):
         return expiration + ".0a1"
--- a/uriloader/moz.build
+++ b/uriloader/moz.build
@@ -1,11 +1,14 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files('**'):
+    BUG_COMPONENT = ('Firefox', 'File Handling')
+
 DIRS += [
     'base',
     'exthandler',
     'prefetch',
 ]
--- a/uriloader/prefetch/moz.build
+++ b/uriloader/prefetch/moz.build
@@ -1,14 +1,17 @@
 # -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
 # vim: set filetype=python:
 # 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/.
 
+with Files("**"):
+    BUG_COMPONENT = ("Core", "Networking: Cache")
+
 XPIDL_SOURCES += [
     'nsIOfflineCacheUpdate.idl',
     'nsIPrefetchService.idl',
 ]
 
 XPIDL_MODULE = 'prefetch'
 
 EXPORTS += [
--- a/xpcom/base/Logging.cpp
+++ b/xpcom/base/Logging.cpp
@@ -277,16 +277,18 @@ public:
     // Can't use rotate at runtime yet.
     MOZ_ASSERT(mRotate == 0, "We don't allow rotate for runtime logfile changes");
     mOutFilePath.reset(strdup(filename));
 
     // Exchange mOutFile and set it to be released once all the writes are done.
     detail::LogFile* newFile = OpenFile(false, 0);
     detail::LogFile* oldFile = mOutFile.exchange(newFile);
 
+    printf("old: %p, new %p\n", newFile, oldFile);
+
     // Since we don't allow changing the logfile if MOZ_LOG_FILE is already set,
     // and we don't allow log rotation when setting it at runtime, mToReleaseFile
     // will be null, so we're not leaking.
     DebugOnly<detail::LogFile*> prevFile = mToReleaseFile.exchange(oldFile);
     MOZ_ASSERT(!prevFile, "Should be null because rotation is not allowed");
 
     // If we just need to release a file, we must force print, in order to
     // trigger the closing and release of mToReleaseFile.
--- a/xpcom/threads/TimerThread.cpp
+++ b/xpcom/threads/TimerThread.cpp
@@ -136,21 +136,17 @@ public:
 // instead of Run().
 class nsTimerEvent final : public CancelableRunnable
 {
 public:
   NS_IMETHOD Run() override;
 
   nsresult Cancel() override
   {
-    // Since nsTimerImpl is not thread-safe, we should release |mTimer|
-    // here in the target thread to avoid race condition. Otherwise,
-    // ~nsTimerEvent() which calls nsTimerImpl::Release() could run in the
-    // timer thread and result in race condition.
-    mTimer = nullptr;
+    mTimer->Cancel();
     return NS_OK;
   }
 
   NS_IMETHOD GetName(nsACString& aName) override;
 
   nsTimerEvent()
     : mTimer()
     , mGeneration(0)
@@ -277,33 +273,26 @@ nsTimerEvent::GetName(nsACString& aName)
 
   mTimer->GetName(aName);
   return NS_OK;
 }
 
 NS_IMETHODIMP
 nsTimerEvent::Run()
 {
-  if (!mTimer) {
-    MOZ_ASSERT(false);
-    return NS_OK;
-  }
-
   if (MOZ_LOG_TEST(GetTimerLog(), LogLevel::Debug)) {
     TimeStamp now = TimeStamp::Now();
     MOZ_LOG(GetTimerLog(), LogLevel::Debug,
            ("[this=%p] time between PostTimerEvent() and Fire(): %fms\n",
             this, (now - mInitTime).ToMilliseconds()));
   }
 
   mTimer->Fire(mGeneration);
 
-  // We call Cancel() to correctly release mTimer.
-  // Read more in the Cancel() implementation.
-  return Cancel();
+  return NS_OK;
 }
 
 nsresult
 TimerThread::Init()
 {
   MOZ_LOG(GetTimerLog(), LogLevel::Debug,
          ("TimerThread::Init [%d]\n", mInitialized));
 
--- a/xpcom/threads/nsTimerImpl.cpp
+++ b/xpcom/threads/nsTimerImpl.cpp
@@ -429,29 +429,34 @@ nsTimerImpl::SetTarget(nsIEventTarget* a
 
 
 void
 nsTimerImpl::Fire(int32_t aGeneration)
 {
   uint8_t oldType;
   uint32_t oldDelay;
   TimeStamp oldTimeout;
+  nsCOMPtr<nsITimer> kungFuDeathGrip;
 
   {
     // Don't fire callbacks or fiddle with refcounts when the mutex is locked.
     // If some other thread Cancels/Inits after this, they're just too late.
     MutexAutoLock lock(mMutex);
     if (aGeneration != mGeneration) {
       return;
     }
 
     mCallbackDuringFire.swap(mCallback);
     oldType = mType;
     oldDelay = mDelay;
     oldTimeout = mTimeout;
+    // Ensure that the nsITimer does not unhook from the nsTimerImpl during
+    // Fire; this will cause null pointer crashes if the user of the timer drops
+    // its reference, and then uses the nsITimer* passed in the callback.
+    kungFuDeathGrip = mITimer;
   }
 
   PROFILER_LABEL("Timer", "Fire",
                  js::ProfileEntry::Category::OTHER);
 
   TimeStamp now = TimeStamp::Now();
   if (MOZ_LOG_TEST(GetTimerLog(), LogLevel::Debug)) {
     TimeDuration   delta = now - oldTimeout;