diff --git a/addon-sdk/source/test/test-clipboard.js b/addon-sdk/source/test/test-clipboard.js
index 358d1b5e0f1..fd1efcaeeb7 100644
--- a/addon-sdk/source/test/test-clipboard.js
+++ b/addon-sdk/source/test/test-clipboard.js
@@ -20,6 +20,10 @@ const base64png = "" +
const { base64jpeg } = require("./fixtures");
+const { platform } = require("sdk/system");
+// For Windows, Mac and Linux, platform returns the following: winnt, darwin and linux.
+var isWindows = platform.toLowerCase().indexOf("win") == 0;
+
const canvasHTML = "data:text/html," + encodeURIComponent(
"\
\
@@ -99,6 +103,13 @@ exports["test With No Flavor"] = function(assert) {
exports["test With Flavor"] = function(assert) {
var contents = "hello there";
var contentsText = "hello there";
+
+ // On windows, HTML clipboard includes extra data.
+ // The values are from widget/windows/nsDataObj.cpp.
+ var contentsWindowsHtml = "\n" +
+ contents +
+ "\n\n";
+
var flavor = "html";
var fullFlavor = "text/html";
var unicodeFlavor = "text";
@@ -110,8 +121,8 @@ exports["test With Flavor"] = function(assert) {
assert.equal(clip.currentFlavors[0], unicodeFlavor);
assert.equal(clip.currentFlavors[1], flavor);
assert.equal(clip.get(), contentsText);
- assert.equal(clip.get(flavor), contents);
- assert.equal(clip.get(fullFlavor), contents);
+ assert.equal(clip.get(flavor), isWindows ? contentsWindowsHtml : contents);
+ assert.equal(clip.get(fullFlavor), isWindows ? contentsWindowsHtml : contents);
assert.equal(clip.get(unicodeFlavor), contentsText);
assert.equal(clip.get(unicodeFullFlavor), contentsText);
};
diff --git a/b2g/config/aries/sources.xml b/b2g/config/aries/sources.xml
index fca1dc32c22..cab76273249 100644
--- a/b2g/config/aries/sources.xml
+++ b/b2g/config/aries/sources.xml
@@ -15,7 +15,7 @@
-
+
@@ -24,7 +24,7 @@
-
+
diff --git a/b2g/config/dolphin/sources.xml b/b2g/config/dolphin/sources.xml
index e904f8d1d70..377fd4b03d2 100644
--- a/b2g/config/dolphin/sources.xml
+++ b/b2g/config/dolphin/sources.xml
@@ -15,7 +15,7 @@
-
+
@@ -24,7 +24,7 @@
-
+
diff --git a/b2g/config/emulator-ics/sources.xml b/b2g/config/emulator-ics/sources.xml
index c015505fdac..7e0a8ec8a86 100644
--- a/b2g/config/emulator-ics/sources.xml
+++ b/b2g/config/emulator-ics/sources.xml
@@ -1,39 +1,48 @@
-
-
-
-
+
+
+
+
+
+
-
-
-
+
+
+
+
-
-
-
-
-
+
+
+
+
-
+
+
+
+
+
+
+
+
+
-
-
+
@@ -42,8 +51,6 @@
-
-
@@ -53,6 +60,7 @@
+
@@ -62,11 +70,9 @@
-
-
@@ -81,9 +87,8 @@
-
-
+
@@ -91,21 +96,17 @@
+
+
+
+
-
-
+
-
-
-
-
-
-
-
diff --git a/b2g/config/emulator-jb/sources.xml b/b2g/config/emulator-jb/sources.xml
index 5dd2a854d7b..801fb2f5702 100644
--- a/b2g/config/emulator-jb/sources.xml
+++ b/b2g/config/emulator-jb/sources.xml
@@ -17,10 +17,10 @@
-
+
-
+
diff --git a/b2g/config/emulator-kk/sources.xml b/b2g/config/emulator-kk/sources.xml
index ac880817d22..a0d798e5aab 100644
--- a/b2g/config/emulator-kk/sources.xml
+++ b/b2g/config/emulator-kk/sources.xml
@@ -15,7 +15,7 @@
-
+
@@ -23,7 +23,7 @@
-
+
diff --git a/b2g/config/emulator-l/sources.xml b/b2g/config/emulator-l/sources.xml
index 9ae39f57799..4934635d19e 100644
--- a/b2g/config/emulator-l/sources.xml
+++ b/b2g/config/emulator-l/sources.xml
@@ -15,7 +15,7 @@
-
+
@@ -23,7 +23,7 @@
-
+
diff --git a/b2g/config/emulator/sources.xml b/b2g/config/emulator/sources.xml
index c015505fdac..7e0a8ec8a86 100644
--- a/b2g/config/emulator/sources.xml
+++ b/b2g/config/emulator/sources.xml
@@ -1,39 +1,48 @@
-
-
-
-
+
+
+
+
+
+
-
-
-
+
+
+
+
-
-
-
-
-
+
+
+
+
-
+
+
+
+
+
+
+
+
+
-
-
+
@@ -42,8 +51,6 @@
-
-
@@ -53,6 +60,7 @@
+
@@ -62,11 +70,9 @@
-
-
@@ -81,9 +87,8 @@
-
-
+
@@ -91,21 +96,17 @@
+
+
+
+
-
-
+
-
-
-
-
-
-
-
diff --git a/b2g/config/flame-kk/sources.xml b/b2g/config/flame-kk/sources.xml
index b6f2a22dbfb..39f8b7948eb 100644
--- a/b2g/config/flame-kk/sources.xml
+++ b/b2g/config/flame-kk/sources.xml
@@ -15,7 +15,7 @@
-
+
@@ -24,7 +24,7 @@
-
+
diff --git a/b2g/config/gaia.json b/b2g/config/gaia.json
index 6e4912fe9d9..4759b67043f 100644
--- a/b2g/config/gaia.json
+++ b/b2g/config/gaia.json
@@ -1,9 +1,9 @@
{
"git": {
- "git_revision": "ea673b5c4cc19c3daca072691a659c68e4c6937f",
+ "git_revision": "37250b125e0db6966875d3b37b117f6d9b76cbc0",
"remote": "https://git.mozilla.org/releases/gaia.git",
"branch": ""
},
- "revision": "f5fc1ed93387c0ff18811fb306e061b8ca77bd40",
+ "revision": "e094d698a05cd04c90815dd11579326f43f02a6b",
"repo_path": "integration/gaia-central"
}
diff --git a/b2g/config/nexus-4-kk/sources.xml b/b2g/config/nexus-4-kk/sources.xml
index 6b3cfe596fe..c0aeb646672 100644
--- a/b2g/config/nexus-4-kk/sources.xml
+++ b/b2g/config/nexus-4-kk/sources.xml
@@ -15,7 +15,7 @@
-
+
@@ -24,7 +24,7 @@
-
+
diff --git a/b2g/config/nexus-4/sources.xml b/b2g/config/nexus-4/sources.xml
index 446c1f6d0b3..f92c433565b 100644
--- a/b2g/config/nexus-4/sources.xml
+++ b/b2g/config/nexus-4/sources.xml
@@ -18,10 +18,10 @@
-
+
-
+
diff --git a/b2g/config/nexus-5-l/sources.xml b/b2g/config/nexus-5-l/sources.xml
index 7f769a41885..85ab233b07e 100644
--- a/b2g/config/nexus-5-l/sources.xml
+++ b/b2g/config/nexus-5-l/sources.xml
@@ -15,7 +15,7 @@
-
+
@@ -24,7 +24,7 @@
-
+
diff --git a/b2g/installer/flash.bat b/b2g/installer/flash.bat
new file mode 100755
index 00000000000..9b5093677de
--- /dev/null
+++ b/b2g/installer/flash.bat
@@ -0,0 +1,73 @@
+@ECHO OFF
+
+REM read config file
+setlocal ENABLEDELAYEDEXPANSION
+set loop=0
+for /F "tokens=*" %%A in (.config) do (
+ SET /A loop=!loop! + 1
+ set %%A
+)
+
+set DEVICE_FOUND=0
+
+REM nexus has device instead of product name
+IF [%PRODUCT_NAME%]==[] (
+ set PRODUCT_NAME=%DEVICE%
+)
+
+REM if nexus 4 assume you are in fastboot mode, can't seem to find drivers
+IF [%DEVICE%]==[mako] (
+call :flash
+)
+
+REM push device from adb to fastboot mode
+win_adb kill-server
+win_adb devices
+win_adb get-state > devicestate.txt
+set /p DEVICE_STATE= < devicestate.txt
+
+IF NOT "%DEVICE_STATE%"=="device" (
+ ECHO Please check :
+ ECHO 1. to make sure that only one device is connected to the computer
+ ECHO 2. the device is turned on with the screen showing
+ ECHO 3. the device is set to debugging via USB : ADB Only or ADB and Devtools
+ ECHO 4. the device drivers are installed on the computer.
+ Del devicestate.txt
+ PAUSE
+ EXIT /b
+)
+
+Del devicestate.txt
+win_adb reboot bootloader
+
+TIMEOUT 5
+
+:flash
+win_fastboot devices 2> fastboot_state.txt
+set /p FASTBOOT_STATE= < fastboot_state.txt
+
+IF NOT [%FASTBOOT_STATE%]==[] (
+ ECHO Please check :
+ ECHO 1. to make sure that only one device is connected to the computer
+ ECHO 2. the device is turned on with an indication that the device is in fastboot mode
+ ECHO 3. the fastboot drivers are installed on the computer.
+ Del fastboot_state.txt
+ PAUSE
+ EXIT /b
+)
+
+Del fastboot_state.txt
+
+ECHO "Flashing build. If nothing mentions that it flashed anything and it looks stuck, make sure you have the drivers installed."
+win_fastboot flash boot out/target/product/%PRODUCT_NAME%/boot.img
+win_fastboot flash system out/target/product/%PRODUCT_NAME%/system.img
+win_fastboot flash persist out/target/product/%PRODUCT_NAME%/persist.img
+win_fastboot flash recovery out/target/product/%PRODUCT_NAME%/recovery.img
+win_fastboot flash cache out/target/product/%PRODUCT_NAME%/cache.img
+win_fastboot flash userdata out/target/product/%PRODUCT_NAME%/userdata.img
+
+ECHO "Done..."
+
+win_fastboot reboot
+echo "Just close the windows as you wish."
+TIMEOUT 5
diff --git a/browser/app/jar.mn b/browser/app/jar.mn
deleted file mode 100644
index cb6ebbfc780..00000000000
--- a/browser/app/jar.mn
+++ /dev/null
@@ -1,3 +0,0 @@
-browser.jar:
-% resource app %
- defaults/permissions (permissions)
diff --git a/browser/app/moz.build b/browser/app/moz.build
index 70b1453e7f5..e10ddf093af 100644
--- a/browser/app/moz.build
+++ b/browser/app/moz.build
@@ -21,6 +21,7 @@ SOURCES += [
FINAL_TARGET_FILES += ['blocklist.xml']
FINAL_TARGET_FILES.defaults.profile += ['profile/prefs.js']
+FINAL_TARGET_FILES.defaults += ['permissions']
DEFINES['APP_VERSION'] = CONFIG['MOZ_APP_VERSION']
@@ -70,7 +71,5 @@ if CONFIG['MOZ_LINKER']:
if CONFIG['HAVE_CLOCK_MONOTONIC']:
OS_LIBS += CONFIG['REALTIME_LIBS']
-JAR_MANIFESTS += ['jar.mn']
-
if CONFIG['GNU_CXX']:
CXXFLAGS += ['-Wshadow']
diff --git a/browser/base/content/blockedSite.xhtml b/browser/base/content/blockedSite.xhtml
index cba04c007c6..8ab5f712258 100644
--- a/browser/base/content/blockedSite.xhtml
+++ b/browser/base/content/blockedSite.xhtml
@@ -22,8 +22,9 @@
+
+
+
diff --git a/dom/canvas/crashtests/crashtests.list b/dom/canvas/crashtests/crashtests.list
index edb54352afe..a8dcd51fa85 100644
--- a/dom/canvas/crashtests/crashtests.list
+++ b/dom/canvas/crashtests/crashtests.list
@@ -23,5 +23,6 @@ load 1099143-1.html
load 1161277-1.html
load 1183363.html
load 1190705.html
+load 1223740-1.html
load 1225381-1.html
load texImage2D.html
diff --git a/dom/canvas/test/mochitest.ini b/dom/canvas/test/mochitest.ini
index 6eb119f9fe6..04c6b1c785c 100644
--- a/dom/canvas/test/mochitest.ini
+++ b/dom/canvas/test/mochitest.ini
@@ -161,6 +161,7 @@ skip-if = toolkit != 'cocoa'
skip-if = toolkit != 'cocoa'
[test_2d.drawImage.zerocanvas.html]
[test_2d.fill.winding.html]
+[test_2d.fill.pattern.imageSmoothingEnabled.html]
# These tests do not pass on any platform; Quartz backend won't pass them
# because we fall back to pixman when one circle doesn't contain the other.
# See bug 512647.
diff --git a/dom/canvas/test/test_2d.fill.pattern.imageSmoothingEnabled.html b/dom/canvas/test/test_2d.fill.pattern.imageSmoothingEnabled.html
new file mode 100644
index 00000000000..8a675f8c0c1
--- /dev/null
+++ b/dom/canvas/test/test_2d.fill.pattern.imageSmoothingEnabled.html
@@ -0,0 +1,81 @@
+
+Canvas test: 2d.fill.pattern.imageSmoothingEnabled
+
+
+
+
+
+
+
+
diff --git a/dom/events/DataTransfer.cpp b/dom/events/DataTransfer.cpp
index 5758ff864a2..76c75197afd 100644
--- a/dom/events/DataTransfer.cpp
+++ b/dom/events/DataTransfer.cpp
@@ -1237,7 +1237,7 @@ DataTransfer::CacheExternalDragFormats()
// there isn't a way to get a list of the formats that might be available on
// all platforms, so just check for the types that can actually be imported
// XXXndeakin there are some other formats but those are platform specific.
- const char* formats[] = { kFileMime, kHTMLMime, kURLMime, kURLDataMime, kUnicodeMime };
+ const char* formats[] = { kFileMime, kHTMLMime, kRTFMime, kURLMime, kURLDataMime, kUnicodeMime };
uint32_t count;
dragSession->GetNumDropItems(&count);
@@ -1279,7 +1279,7 @@ DataTransfer::CacheExternalClipboardFormats()
// there isn't a way to get a list of the formats that might be available on
// all platforms, so just check for the types that can actually be imported
- const char* formats[] = { kFileMime, kHTMLMime, kURLMime, kURLDataMime, kUnicodeMime };
+ const char* formats[] = { kFileMime, kHTMLMime, kRTFMime, kURLMime, kURLDataMime, kUnicodeMime };
for (uint32_t f = 0; f < mozilla::ArrayLength(formats); ++f) {
// check each format one at a time
@@ -1361,7 +1361,14 @@ DataTransfer::FillInExternalData(TransferItem& aItem, uint32_t aIndex)
variant->SetAsAString(str);
}
else {
- variant->SetAsISupports(data);
+ nsCOMPtr supportscstr = do_QueryInterface(data);
+ if (supportscstr) {
+ nsAutoCString str;
+ supportscstr->GetData(str);
+ variant->SetAsACString(str);
+ } else {
+ variant->SetAsISupports(data);
+ }
}
aItem.mData = variant;
diff --git a/dom/events/EventStateManager.cpp b/dom/events/EventStateManager.cpp
index cb18df98fbb..5199be81e1c 100644
--- a/dom/events/EventStateManager.cpp
+++ b/dom/events/EventStateManager.cpp
@@ -288,6 +288,7 @@ EventStateManager::DeltaAccumulator*
EventStateManager::EventStateManager()
: mLockCursor(0)
+ , mLastFrameConsumedSetCursor(false)
, mPreLockPoint(0,0)
, mCurrentTarget(nullptr)
// init d&d gesture state machine variables
@@ -3527,8 +3528,19 @@ EventStateManager::UpdateCursor(nsPresContext* aPresContext,
nsIFrame::Cursor framecursor;
nsPoint pt = nsLayoutUtils::GetEventCoordinatesRelativeTo(aEvent,
aTargetFrame);
- if (NS_FAILED(aTargetFrame->GetCursor(pt, framecursor)))
- return; // don't update the cursor if we failed to get it from the frame see bug 118877
+ // Avoid setting cursor when the mouse is over a windowless pluign.
+ if (NS_FAILED(aTargetFrame->GetCursor(pt, framecursor))) {
+ if (XRE_IsContentProcess()) {
+ mLastFrameConsumedSetCursor = true;
+ }
+ return;
+ }
+ // Make sure cursors get reset after the mouse leaves a
+ // windowless plugin frame.
+ if (mLastFrameConsumedSetCursor) {
+ ClearCachedWidgetCursor(aTargetFrame);
+ mLastFrameConsumedSetCursor = false;
+ }
cursor = framecursor.mCursor;
container = framecursor.mContainer;
haveHotspot = framecursor.mHaveHotspot;
diff --git a/dom/events/EventStateManager.h b/dom/events/EventStateManager.h
index c6a83fbfbcf..0575233c05e 100644
--- a/dom/events/EventStateManager.h
+++ b/dom/events/EventStateManager.h
@@ -861,6 +861,7 @@ private:
bool dispatchedToContentProcess);
int32_t mLockCursor;
+ bool mLastFrameConsumedSetCursor;
// Last mouse event refPoint (the offset from the widget's origin in
// device pixels) when mouse was locked, used to restore mouse position
diff --git a/dom/events/IMEContentObserver.cpp b/dom/events/IMEContentObserver.cpp
index e744614cd11..9fa14f63878 100644
--- a/dom/events/IMEContentObserver.cpp
+++ b/dom/events/IMEContentObserver.cpp
@@ -40,7 +40,7 @@ namespace mozilla {
using namespace widget;
-PRLogModuleInfo* sIMECOLog = nullptr;
+LazyLogModule sIMECOLog("IMEContentObserver");
static const char*
ToChar(bool aBool)
@@ -207,9 +207,6 @@ IMEContentObserver::IMEContentObserver()
#ifdef DEBUG
mTextChangeData.Test();
#endif
- if (!sIMECOLog) {
- sIMECOLog = PR_NewLogModule("IMEContentObserver");
- }
}
void
diff --git a/dom/events/IMEStateManager.cpp b/dom/events/IMEStateManager.cpp
index 8fd2a47a9a6..acada04c2de 100644
--- a/dom/events/IMEStateManager.cpp
+++ b/dom/events/IMEStateManager.cpp
@@ -57,7 +57,7 @@ using namespace widget;
* for debug, log the information with LogLevel::Debug. In this case, the log
* should start with "ISM: (),".
*/
-PRLogModuleInfo* sISMLog = nullptr;
+LazyLogModule sISMLog("IMEStateManager");
static const char*
GetBoolName(bool aBool)
@@ -175,10 +175,6 @@ bool IMEStateManager::sRemoteHasFocus = false;
void
IMEStateManager::Init()
{
- if (!sISMLog) {
- sISMLog = PR_NewLogModule("IMEStateManager");
- }
-
Preferences::AddBoolVarCache(
&sCheckForIMEUnawareWebApps,
"intl.ime.hack.on_ime_unaware_apps.fire_key_events_for_composition",
diff --git a/dom/fetch/Fetch.cpp b/dom/fetch/Fetch.cpp
index 3484a2b9d70..af11554ff94 100644
--- a/dom/fetch/Fetch.cpp
+++ b/dom/fetch/Fetch.cpp
@@ -976,8 +976,14 @@ FetchBody::ContinueConsumeBody(nsresult aStatus, uint32_t aResultLength
// FetchBody on the main thread.
RefPtr> r =
new CancelPumpRunnable(this);
- if (!r->Dispatch(mWorkerPrivate->GetJSContext())) {
+ ErrorResult rv;
+ r->Dispatch(rv);
+ if (rv.Failed()) {
NS_WARNING("Could not dispatch CancelPumpRunnable. Nothing we can do here");
+ // None of our callers are callled directly from JS, so there is no
+ // point in trying to propagate this failure out of here. And
+ // localPromise is already rejected. Just suppress the failure.
+ rv.SuppressException();
}
}
}
diff --git a/dom/fetch/FetchDriver.cpp b/dom/fetch/FetchDriver.cpp
index 50c31bec433..6c1fa9e8a7d 100644
--- a/dom/fetch/FetchDriver.cpp
+++ b/dom/fetch/FetchDriver.cpp
@@ -29,6 +29,7 @@
#include "nsPrintfCString.h"
#include "nsStreamUtils.h"
#include "nsStringStream.h"
+#include "nsHttpChannel.h"
#include "mozilla/dom/File.h"
#include "mozilla/dom/workers/Workers.h"
@@ -51,7 +52,6 @@ FetchDriver::FetchDriver(InternalRequest* aRequest, nsIPrincipal* aPrincipal,
, mLoadGroup(aLoadGroup)
, mRequest(aRequest)
, mHasBeenCrossSite(false)
- , mFoundOpaqueRedirect(false)
, mResponseAvailableCalled(false)
, mFetchCalled(false)
{
@@ -236,6 +236,10 @@ FetchDriver::HttpFetch()
return NS_ERROR_UNEXPECTED;
}
+ if (mRequest->GetRedirectMode() != RequestRedirect::Follow) {
+ secFlags |= nsILoadInfo::SEC_DONT_FOLLOW_REDIRECTS;
+ }
+
// From here on we create a channel and set its properties with the
// information from the InternalRequest. This is an implementation detail.
MOZ_ASSERT(mLoadGroup);
@@ -441,7 +445,9 @@ FetchDriver::IsUnsafeRequest()
}
already_AddRefed
-FetchDriver::BeginAndGetFilteredResponse(InternalResponse* aResponse, nsIURI* aFinalURI)
+FetchDriver::BeginAndGetFilteredResponse(InternalResponse* aResponse,
+ nsIURI* aFinalURI,
+ bool aFoundOpaqueRedirect)
{
MOZ_ASSERT(aResponse);
nsAutoCString reqURL;
@@ -454,7 +460,7 @@ FetchDriver::BeginAndGetFilteredResponse(InternalResponse* aResponse, nsIURI* aF
MOZ_ASSERT(NS_SUCCEEDED(rv));
RefPtr filteredResponse;
- if (mFoundOpaqueRedirect) {
+ if (aFoundOpaqueRedirect) {
filteredResponse = aResponse->OpaqueRedirectResponse();
} else {
switch (mRequest->GetResponseTainting()) {
@@ -551,10 +557,22 @@ FetchDriver::OnStartRequest(nsIRequest* aRequest,
nsCOMPtr httpChannel = do_QueryInterface(aRequest);
nsCOMPtr jarChannel = do_QueryInterface(aRequest);
+ bool foundOpaqueRedirect = false;
+
if (httpChannel) {
uint32_t responseStatus;
httpChannel->GetResponseStatus(&responseStatus);
+ if (mozilla::net::nsHttpChannel::IsRedirectStatus(responseStatus)) {
+ if (mRequest->GetRedirectMode() == RequestRedirect::Error) {
+ FailWithNetworkError();
+ return NS_BINDING_FAILED;
+ }
+ if (mRequest->GetRedirectMode() == RequestRedirect::Manual) {
+ foundOpaqueRedirect = true;
+ }
+ }
+
nsAutoCString statusText;
httpChannel->GetResponseStatusText(statusText);
@@ -660,7 +678,8 @@ FetchDriver::OnStartRequest(nsIRequest* aRequest,
// Resolves fetch() promise which may trigger code running in a worker. Make
// sure the Response is fully initialized before calling this.
- mResponse = BeginAndGetFilteredResponse(response, channelURI);
+ mResponse = BeginAndGetFilteredResponse(response, channelURI,
+ foundOpaqueRedirect);
nsCOMPtr sts = do_GetService(NS_STREAMTRANSPORTSERVICE_CONTRACTID, &rv);
if (NS_WARN_IF(NS_FAILED(rv))) {
@@ -743,6 +762,12 @@ FetchDriver::AsyncOnChannelRedirect(nsIChannel* aOldChannel,
return NS_OK;
}
+ // We should only ever get here if we use a "follow" redirect policy,
+ // or if if we set an "error" policy as a result of a CORS policy.
+ MOZ_ASSERT(mRequest->GetRedirectMode() == RequestRedirect::Follow ||
+ (mRequest->GetRedirectMode() == RequestRedirect::Error &&
+ IsUnsafeRequest()));
+
// HTTP Fetch step 5, "redirect status", step 1
if (NS_WARN_IF(mRequest->GetRedirectMode() == RequestRedirect::Error)) {
aOldChannel->Cancel(NS_BINDING_FAILED);
@@ -763,33 +788,10 @@ FetchDriver::AsyncOnChannelRedirect(nsIChannel* aOldChannel,
// HTTP Fetch step 5, "redirect status", step 10 requires us to halt the
// redirect, but successfully return an opaqueredirect Response to the
// initiating Fetch.
- if (mRequest->GetRedirectMode() == RequestRedirect::Manual) {
- // Ideally we would simply not cancel the old channel and allow it to
- // be processed as normal. Unfortunately this is quite fragile and
- // other redirect handlers can easily break it for certain use cases.
- //
- // For example, nsCORSListenerProxy cancels vetoed redirect channels.
- // The HTTP cache will also error on vetoed redirects when the
- // redirect has been previously cached.
- //
- // Therefore simulate the completion of the channel to produce the
- // opaqueredirect Response and then cancel the original channel. This
- // will result in OnStartRequest() getting called twice, but the second
- // time will be with an error response (from the Cancel) which will
- // be ignored.
- MOZ_ASSERT(!mFoundOpaqueRedirect);
- mFoundOpaqueRedirect = true;
- Unused << OnStartRequest(aOldChannel, nullptr);
- Unused << OnStopRequest(aOldChannel, nullptr, NS_OK);
-
- aOldChannel->Cancel(NS_BINDING_FAILED);
-
- return NS_BINDING_FAILED;
- }
// The following steps are from HTTP Fetch step 5, "redirect status", step 11
- // which requires the RequestRedirect to be "follow".
- MOZ_ASSERT(mRequest->GetRedirectMode() == RequestRedirect::Follow);
+ // which requires the RequestRedirect to be "follow". We asserted that we're
+ // in either "follow" or "error" mode here.
// HTTP Fetch step 5, "redirect status", steps 11.1 and 11.2 block redirecting
// to a URL with credentials in CORS mode. This is implemented in
diff --git a/dom/fetch/FetchDriver.h b/dom/fetch/FetchDriver.h
index ee8f2d0347a..8071316a6f8 100644
--- a/dom/fetch/FetchDriver.h
+++ b/dom/fetch/FetchDriver.h
@@ -83,7 +83,6 @@ private:
RefPtr mObserver;
nsCOMPtr mDocument;
bool mHasBeenCrossSite;
- bool mFoundOpaqueRedirect;
DebugOnly mResponseAvailableCalled;
DebugOnly mFetchCalled;
@@ -100,7 +99,8 @@ private:
// Returns the filtered response sent to the observer.
// Callers who don't have access to a channel can pass null for aFinalURI.
already_AddRefed
- BeginAndGetFilteredResponse(InternalResponse* aResponse, nsIURI* aFinalURI);
+ BeginAndGetFilteredResponse(InternalResponse* aResponse, nsIURI* aFinalURI,
+ bool aFoundOpaqueRedirect);
// Utility since not all cases need to do any post processing of the filtered
// response.
nsresult FailWithNetworkError();
diff --git a/dom/html/HTMLAnchorElement.cpp b/dom/html/HTMLAnchorElement.cpp
index 7115f14e678..91bbe3ad2a4 100644
--- a/dom/html/HTMLAnchorElement.cpp
+++ b/dom/html/HTMLAnchorElement.cpp
@@ -63,13 +63,11 @@ NS_IMPL_CYCLE_COLLECTION_CLASS(HTMLAnchorElement)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(HTMLAnchorElement,
nsGenericHTMLElement)
- tmp->Link::Traverse(cb);
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRelList)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(HTMLAnchorElement,
nsGenericHTMLElement)
- tmp->Link::Unlink();
NS_IMPL_CYCLE_COLLECTION_UNLINK(mRelList)
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
@@ -324,17 +322,13 @@ HTMLAnchorElement::RelList()
NS_IMETHODIMP \
HTMLAnchorElement::Get##_part(nsAString& a##_part) \
{ \
- ErrorResult rv; \
- Link::Get##_part(a##_part, rv); \
- MOZ_ASSERT(!rv.Failed()); \
+ Link::Get##_part(a##_part); \
return NS_OK; \
} \
NS_IMETHODIMP \
HTMLAnchorElement::Set##_part(const nsAString& a##_part) \
{ \
- ErrorResult rv; \
- Link::Set##_part(a##_part, rv); \
- MOZ_ASSERT(!rv.Failed()); \
+ Link::Set##_part(a##_part); \
return NS_OK; \
}
diff --git a/dom/html/HTMLAnchorElement.h b/dom/html/HTMLAnchorElement.h
index e3cf409c694..e29d6d8b1f9 100644
--- a/dom/html/HTMLAnchorElement.h
+++ b/dom/html/HTMLAnchorElement.h
@@ -89,10 +89,8 @@ public:
virtual bool HasDeferredDNSPrefetchRequest() override;
// WebIDL API
- void GetHref(nsAString& aValue, ErrorResult& rv)
- {
- GetHTMLURIAttr(nsGkAtoms::href, aValue);
- }
+
+ // The XPCOM GetHref is OK for us
void SetHref(const nsAString& aValue, mozilla::ErrorResult& rv)
{
SetHTMLAttr(nsGkAtoms::href, aValue, rv);
@@ -156,8 +154,8 @@ public:
// Link::GetOrigin is OK for us
- using Link::GetProtocol;
- using Link::SetProtocol;
+ // Link::GetProtocol is OK for us
+ // Link::SetProtocol is OK for us
// Link::GetUsername is OK for us
// Link::SetUsername is OK for us
@@ -165,23 +163,23 @@ public:
// Link::GetPassword is OK for us
// Link::SetPassword is OK for us
- using Link::GetHost;
- using Link::SetHost;
+ // Link::Link::GetHost is OK for us
+ // Link::Link::SetHost is OK for us
- using Link::GetHostname;
- using Link::SetHostname;
+ // Link::Link::GetHostname is OK for us
+ // Link::Link::SetHostname is OK for us
- using Link::GetPort;
- using Link::SetPort;
+ // Link::Link::GetPort is OK for us
+ // Link::Link::SetPort is OK for us
- using Link::GetPathname;
- using Link::SetPathname;
+ // Link::Link::GetPathname is OK for us
+ // Link::Link::SetPathname is OK for us
- using Link::GetSearch;
- using Link::SetSearch;
+ // Link::Link::GetSearch is OK for us
+ // Link::Link::SetSearch is OK for us
- using Link::GetHash;
- using Link::SetHash;
+ // Link::Link::GetHash is OK for us
+ // Link::Link::SetHash is OK for us
// The XPCOM URI decomposition attributes are fine for us
void GetCoords(DOMString& aValue)
@@ -224,9 +222,9 @@ public:
{
SetHTMLAttr(nsGkAtoms::shape, aValue, rv);
}
- void Stringify(nsAString& aResult, ErrorResult& aError)
+ void Stringify(nsAString& aResult)
{
- GetHref(aResult, aError);
+ GetHref(aResult);
}
protected:
diff --git a/dom/html/HTMLAreaElement.cpp b/dom/html/HTMLAreaElement.cpp
index b2f8a2a47b8..eecdf27317a 100644
--- a/dom/html/HTMLAreaElement.cpp
+++ b/dom/html/HTMLAreaElement.cpp
@@ -40,13 +40,11 @@ NS_IMPL_CYCLE_COLLECTION_CLASS(HTMLAreaElement)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(HTMLAreaElement,
nsGenericHTMLElement)
- tmp->Link::Traverse(cb);
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRelList)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(HTMLAreaElement,
nsGenericHTMLElement)
- tmp->Link::Unlink();
NS_IMPL_CYCLE_COLLECTION_UNLINK(mRelList)
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
@@ -207,17 +205,13 @@ HTMLAreaElement::UnsetAttr(int32_t aNameSpaceID, nsIAtom* aAttribute,
NS_IMETHODIMP \
HTMLAreaElement::Get##_part(nsAString& a##_part) \
{ \
- ErrorResult rv; \
- Link::Get##_part(a##_part, rv); \
- MOZ_ASSERT(!rv.Failed()); \
+ Link::Get##_part(a##_part); \
return NS_OK; \
} \
NS_IMETHODIMP \
HTMLAreaElement::Set##_part(const nsAString& a##_part) \
{ \
- ErrorResult rv; \
- Link::Set##_part(a##_part, rv); \
- MOZ_ASSERT(!rv.Failed()); \
+ Link::Set##_part(a##_part); \
return NS_OK; \
}
diff --git a/dom/html/HTMLAreaElement.h b/dom/html/HTMLAreaElement.h
index bd876ec2afc..c8dadd6af70 100644
--- a/dom/html/HTMLAreaElement.h
+++ b/dom/html/HTMLAreaElement.h
@@ -90,10 +90,7 @@ public:
SetHTMLAttr(nsGkAtoms::shape, aShape, aError);
}
- void GetHref(nsAString& aHref, ErrorResult& aError)
- {
- aError = GetHref(aHref);
- }
+ // The XPCOM GetHref is OK for us
void SetHref(const nsAString& aHref, ErrorResult& aError)
{
aError = SetHref(aHref);
@@ -139,8 +136,8 @@ public:
// The Link::GetOrigin is OK for us
- using Link::GetProtocol;
- using Link::SetProtocol;
+ // Link::Link::GetProtocol is OK for us
+ // Link::Link::SetProtocol is OK for us
// The Link::GetUsername is OK for us
// The Link::SetUsername is OK for us
@@ -148,23 +145,23 @@ public:
// The Link::GetPassword is OK for us
// The Link::SetPassword is OK for us
- using Link::GetHost;
- using Link::SetHost;
+ // Link::Link::GetHost is OK for us
+ // Link::Link::SetHost is OK for us
- using Link::GetHostname;
- using Link::SetHostname;
+ // Link::Link::GetHostname is OK for us
+ // Link::Link::SetHostname is OK for us
- using Link::GetPort;
- using Link::SetPort;
+ // Link::Link::GetPort is OK for us
+ // Link::Link::SetPort is OK for us
- using Link::GetPathname;
- using Link::SetPathname;
+ // Link::Link::GetPathname is OK for us
+ // Link::Link::SetPathname is OK for us
- using Link::GetSearch;
- using Link::SetSearch;
+ // Link::Link::GetSearch is OK for us
+ // Link::Link::SetSearch is OK for us
- using Link::GetHash;
- using Link::SetHash;
+ // Link::Link::GetHash is OK for us
+ // Link::Link::SetHash is OK for us
// The Link::GetSearchParams is OK for us
@@ -178,9 +175,9 @@ public:
SetHTMLBoolAttr(nsGkAtoms::nohref, aValue, aError);
}
- void Stringify(nsAString& aResult, ErrorResult& aError)
+ void Stringify(nsAString& aResult)
{
- GetHref(aResult, aError);
+ GetHref(aResult);
}
protected:
diff --git a/dom/html/HTMLImageElement.cpp b/dom/html/HTMLImageElement.cpp
index f3505766a67..b339eab749c 100644
--- a/dom/html/HTMLImageElement.cpp
+++ b/dom/html/HTMLImageElement.cpp
@@ -535,12 +535,6 @@ HTMLImageElement::SetAttr(int32_t aNameSpaceID, nsIAtom* aName,
if (aNameSpaceID == kNameSpaceID_None &&
aName == nsGkAtoms::src) {
- // This is for dom.disable_image_src_set, which predates "srcset"
- // as an attribute. See Bug 773429
- if (nsContentUtils::IsImageSrcSetDisabled()) {
- return NS_OK;
- }
-
if (InResponsiveMode()) {
if (mResponsiveSelector &&
mResponsiveSelector->Content() == this) {
diff --git a/dom/html/HTMLLinkElement.cpp b/dom/html/HTMLLinkElement.cpp
index 97e7f83393f..ddaab02560b 100644
--- a/dom/html/HTMLLinkElement.cpp
+++ b/dom/html/HTMLLinkElement.cpp
@@ -48,7 +48,6 @@ NS_IMPL_CYCLE_COLLECTION_CLASS(HTMLLinkElement)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN_INHERITED(HTMLLinkElement,
nsGenericHTMLElement)
tmp->nsStyleLinkElement::Traverse(cb);
- tmp->Link::Traverse(cb);
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mRelList)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mImportLoader)
NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
@@ -56,7 +55,6 @@ NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN_INHERITED(HTMLLinkElement,
nsGenericHTMLElement)
tmp->nsStyleLinkElement::Unlink();
- tmp->Link::Unlink();
NS_IMPL_CYCLE_COLLECTION_UNLINK(mRelList)
NS_IMPL_CYCLE_COLLECTION_UNLINK(mImportLoader)
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
diff --git a/dom/html/HTMLMediaElement.cpp b/dom/html/HTMLMediaElement.cpp
index 06df311970e..15687b9a5e2 100644
--- a/dom/html/HTMLMediaElement.cpp
+++ b/dom/html/HTMLMediaElement.cpp
@@ -91,8 +91,9 @@
#include
#include
-static PRLogModuleInfo* gMediaElementLog;
-static PRLogModuleInfo* gMediaElementEventsLog;
+static mozilla::LazyLogModule gMediaElementLog("nsMediaElement");
+static mozilla::LazyLogModule gMediaElementEventsLog("nsMediaElementEvents");
+
#define LOG(type, msg) MOZ_LOG(gMediaElementLog, type, msg)
#define LOG_EVENT(type, msg) MOZ_LOG(gMediaElementEventsLog, type, msg)
@@ -107,12 +108,6 @@ static PRLogModuleInfo* gMediaElementEventsLog;
#include "mozilla/EventStateManager.h"
-#if defined(MOZ_B2G) && !defined(MOZ_GRAPHENE)
-// This controls the b2g specific of pausing the media element when the
-// AudioChannel tells us to mute it.
-#define PAUSE_MEDIA_ELEMENT_FROM_AUDIOCHANNEL
-#endif
-
using namespace mozilla::layers;
using mozilla::net::nsMediaFragmentURIParser;
@@ -578,6 +573,7 @@ void
HTMLMediaElement::SetSrcObject(DOMMediaStream* aValue)
{
mSrcAttrStream = aValue;
+ UpdateAudioChannelPlayingState();
DoLoad();
}
@@ -602,6 +598,7 @@ void
HTMLMediaElement::SetMozSrcObject(DOMMediaStream* aValue)
{
mSrcAttrStream = aValue;
+ UpdateAudioChannelPlayingState();
DoLoad();
}
@@ -768,6 +765,7 @@ void HTMLMediaElement::AbortExistingLoads()
FireTimeUpdate(false);
}
DispatchAsyncEvent(NS_LITERAL_STRING("emptied"));
+ UpdateAudioChannelPlayingState();
}
// We may have changed mPaused, mAutoplaying, and other
@@ -1711,6 +1709,7 @@ HTMLMediaElement::Pause(ErrorResult& aRv)
// We changed mPaused and mAutoplaying which can affect AddRemoveSelfReference
AddRemoveSelfReference();
UpdateSrcMediaStreamPlaying();
+ UpdateAudioChannelPlayingState();
if (!oldPaused) {
FireTimeUpdate(false);
@@ -2113,13 +2112,6 @@ HTMLMediaElement::HTMLMediaElement(already_AddRefed& aNo
mFirstFrameLoaded(false),
mDefaultPlaybackStartPosition(0.0)
{
- if (!gMediaElementLog) {
- gMediaElementLog = PR_NewLogModule("nsMediaElement");
- }
- if (!gMediaElementEventsLog) {
- gMediaElementEventsLog = PR_NewLogModule("nsMediaElementEvents");
- }
-
mAudioChannel = AudioChannelService::GetDefaultAudioChannel();
mPaused.SetOuter(this);
@@ -2317,6 +2309,7 @@ HTMLMediaElement::PlayInternal(bool aCallerIsChrome)
AddRemoveSelfReference();
UpdatePreloadAction();
UpdateSrcMediaStreamPlaying();
+ UpdateAudioChannelPlayingState();
return NS_OK;
}
@@ -2897,6 +2890,7 @@ nsresult HTMLMediaElement::FinishDecoderSetup(MediaDecoder* aDecoder,
// We may want to suspend the new stream now.
// This will also do an AddRemoveSelfReference.
NotifyOwnerDocumentActivityChangedInternal();
+ UpdateAudioChannelPlayingState();
if (!mPaused) {
SetPlayedOrSeeked(true);
@@ -3993,6 +3987,7 @@ void HTMLMediaElement::CheckAutoplayDataReady()
// We changed mPaused which can affect AddRemoveSelfReference
AddRemoveSelfReference();
UpdateSrcMediaStreamPlaying();
+ UpdateAudioChannelPlayingState();
if (mDecoder) {
SetPlayedOrSeeked(true);
@@ -4007,13 +4002,13 @@ void HTMLMediaElement::CheckAutoplayDataReady()
}
-bool HTMLMediaElement::IsActive()
+bool HTMLMediaElement::IsActive() const
{
nsIDocument* ownerDoc = OwnerDoc();
return ownerDoc && ownerDoc->IsActive() && ownerDoc->IsVisible();
}
-bool HTMLMediaElement::IsHidden()
+bool HTMLMediaElement::IsHidden() const
{
if (mElementInTreeState == ELEMENT_NOT_INTREE_HAD_INTREE) {
return true;
@@ -4192,6 +4187,7 @@ void HTMLMediaElement::SuspendOrResumeElement(bool aPauseElement, bool aSuspendE
if (aPauseElement != mPausedForInactiveDocumentOrChannel) {
mPausedForInactiveDocumentOrChannel = aPauseElement;
UpdateSrcMediaStreamPlaying();
+ UpdateAudioChannelPlayingState();
if (aPauseElement) {
if (mMediaSource) {
ReportMSETelemetry();
@@ -4251,15 +4247,12 @@ bool HTMLMediaElement::IsBeingDestroyed()
void HTMLMediaElement::NotifyOwnerDocumentActivityChanged()
{
bool pauseElement = NotifyOwnerDocumentActivityChangedInternal();
- if (pauseElement && mAudioChannelAgent
-#ifdef PAUSE_MEDIA_ELEMENT_FROM_AUDIOCHANNEL
+ if (pauseElement && mAudioChannelAgent &&
// On B2G, NotifyOwnerDocumentActivityChangedInternal may return true for
// two reasons: the document no longer being active, or the element being
// paused by the audio channel. However we are only interested in the
// first case here, so we need to filter out the second case.
- && !ComputedMuted()
-#endif
- ) {
+ (!UseAudioChannelAPI() || !ComputedMuted())) {
// If the element is being paused since we are navigating away from the
// document, notify the audio channel agent.
// Be careful to ignore this event during a docshell frame swap.
@@ -4283,15 +4276,13 @@ HTMLMediaElement::NotifyOwnerDocumentActivityChangedInternal()
}
bool pauseElement = !IsActive();
-#ifdef PAUSE_MEDIA_ELEMENT_FROM_AUDIOCHANNEL
// Only pause the element when we start playing. If we pause without playing
// audio, the resource loading would be affected unexpectedly. For example,
// the media element is muted by default, but we don't want this behavior
// interrupting the loading process.
- if (mAudioChannelAgent) {
+ if (UseAudioChannelAPI() && mAudioChannelAgent) {
pauseElement |= ComputedMuted();
}
-#endif
SuspendOrResumeElement(pauseElement, !IsActive());
@@ -4725,9 +4716,10 @@ nsresult HTMLMediaElement::UpdateChannelMuteState(float aVolume, bool aMuted)
}
}
-#ifdef PAUSE_MEDIA_ELEMENT_FROM_AUDIOCHANNEL
- SuspendOrResumeElement(ComputedMuted(), false);
-#endif
+ if (UseAudioChannelAPI()) {
+ SuspendOrResumeElement(ComputedMuted(), false);
+ }
+
return NS_OK;
}
@@ -4761,6 +4753,11 @@ HTMLMediaElement::IsPlayingThroughTheAudioChannel() const
return false;
}
+ // We should consider any bfcached page or inactive document as non-playing.
+ if (!IsActive()) {
+ return false;
+ }
+
// A loop always is playing
if (HasAttr(kNameSpaceID_None, nsGkAtoms::loop)) {
return true;
@@ -4837,9 +4834,9 @@ NS_IMETHODIMP HTMLMediaElement::WindowVolumeChanged(float aVolume, bool aMuted)
UpdateChannelMuteState(aVolume, aMuted);
-#ifdef PAUSE_MEDIA_ELEMENT_FROM_AUDIOCHANNEL
- mPaused.SetCanPlay(!aMuted);
-#endif
+ if (UseAudioChannelAPI()) {
+ mPaused.SetCanPlay(!aMuted);
+ }
return NS_OK;
}
diff --git a/dom/html/HTMLMediaElement.h b/dom/html/HTMLMediaElement.h
index b9c376e108a..1935c949f13 100644
--- a/dom/html/HTMLMediaElement.h
+++ b/dom/html/HTMLMediaElement.h
@@ -214,9 +214,9 @@ public:
// suspended the channel.
virtual void NotifySuspendedByCache(bool aIsSuspended) final override;
- virtual bool IsActive() final override;
+ virtual bool IsActive() const final override;
- virtual bool IsHidden() final override;
+ virtual bool IsHidden() const final override;
// In order to create overlayImageContainer to support DOMHwMediaStream.
VideoFrameContainer* GetOverlayImageVideoFrameContainer();
diff --git a/dom/html/HTMLTrackElement.cpp b/dom/html/HTMLTrackElement.cpp
index abdda2fa8c2..6a47128dfd0 100644
--- a/dom/html/HTMLTrackElement.cpp
+++ b/dom/html/HTMLTrackElement.cpp
@@ -40,7 +40,7 @@
#include "nsThreadUtils.h"
#include "nsVideoFrame.h"
-static PRLogModuleInfo* gTrackElementLog;
+static mozilla::LazyLogModule gTrackElementLog("nsTrackElement");
#define LOG(type, msg) MOZ_LOG(gTrackElementLog, type, msg)
// Replace the usual NS_IMPL_NS_NEW_HTML_ELEMENT(Track) so
@@ -76,9 +76,6 @@ static MOZ_CONSTEXPR const char* kKindTableDefaultString = kKindTable[0].tag;
HTMLTrackElement::HTMLTrackElement(already_AddRefed& aNodeInfo)
: nsGenericHTMLElement(aNodeInfo)
{
- if (!gTrackElementLog) {
- gTrackElementLog = PR_NewLogModule("nsTrackElement");
- }
}
HTMLTrackElement::~HTMLTrackElement()
diff --git a/dom/html/nsHTMLDNSPrefetch.cpp b/dom/html/nsHTMLDNSPrefetch.cpp
index 09d281b9bec..1c36218f51b 100644
--- a/dom/html/nsHTMLDNSPrefetch.cpp
+++ b/dom/html/nsHTMLDNSPrefetch.cpp
@@ -182,8 +182,7 @@ nsHTMLDNSPrefetch::CancelPrefetch(Link *aElement,
return NS_ERROR_NOT_AVAILABLE;
nsAutoString hostname;
- ErrorResult rv;
- aElement->GetHostname(hostname, rv);
+ aElement->GetHostname(hostname);
return CancelPrefetch(hostname, flags, aReason);
}
diff --git a/dom/indexedDB/ActorsParent.cpp b/dom/indexedDB/ActorsParent.cpp
index 53e42e034c9..8250c9b6d58 100644
--- a/dom/indexedDB/ActorsParent.cpp
+++ b/dom/indexedDB/ActorsParent.cpp
@@ -20994,80 +20994,33 @@ OpenDatabaseOp::MetadataToSpec(DatabaseSpec& aSpec)
AssertIsOnOwningThread();
MOZ_ASSERT(mMetadata);
- class MOZ_STACK_CLASS Helper final
- {
- DatabaseSpec& mSpec;
- ObjectStoreSpec* mCurrentObjectStoreSpec;
+ aSpec.metadata() = mMetadata->mCommonMetadata;
- public:
- static void
- CopyToSpec(const FullDatabaseMetadata* aMetadata, DatabaseSpec& aSpec)
- {
- AssertIsOnBackgroundThread();
- MOZ_ASSERT(aMetadata);
+ for (auto objectStoreIter = mMetadata->mObjectStores.ConstIter();
+ !objectStoreIter.Done();
+ objectStoreIter.Next()) {
+ FullObjectStoreMetadata* metadata = objectStoreIter.UserData();
+ MOZ_ASSERT(objectStoreIter.Key());
+ MOZ_ASSERT(metadata);
- aSpec.metadata() = aMetadata->mCommonMetadata;
+ // XXX This should really be fallible...
+ ObjectStoreSpec* objectStoreSpec = aSpec.objectStores().AppendElement();
+ objectStoreSpec->metadata() = metadata->mCommonMetadata;
- Helper helper(aSpec);
- aMetadata->mObjectStores.EnumerateRead(Enumerate, &helper);
- }
-
- private:
- explicit Helper(DatabaseSpec& aSpec)
- : mSpec(aSpec)
- , mCurrentObjectStoreSpec(nullptr)
- { }
-
- static PLDHashOperator
- Enumerate(const uint64_t& aKey,
- FullObjectStoreMetadata* aValue,
- void* aClosure)
- {
- MOZ_ASSERT(aKey);
- MOZ_ASSERT(aValue);
- MOZ_ASSERT(aClosure);
-
- auto* helper = static_cast(aClosure);
-
- MOZ_ASSERT(!helper->mCurrentObjectStoreSpec);
+ for (auto indexIter = metadata->mIndexes.Iter();
+ !indexIter.Done();
+ indexIter.Next()) {
+ FullIndexMetadata* indexMetadata = indexIter.UserData();
+ MOZ_ASSERT(indexIter.Key());
+ MOZ_ASSERT(indexMetadata);
// XXX This should really be fallible...
- ObjectStoreSpec* objectStoreSpec =
- helper->mSpec.objectStores().AppendElement();
- objectStoreSpec->metadata() = aValue->mCommonMetadata;
-
- AutoRestore ar(helper->mCurrentObjectStoreSpec);
- helper->mCurrentObjectStoreSpec = objectStoreSpec;
-
- aValue->mIndexes.EnumerateRead(Enumerate, helper);
-
- return PL_DHASH_NEXT;
+ IndexMetadata* metadata = objectStoreSpec->indexes().AppendElement();
+ *metadata = indexMetadata->mCommonMetadata;
}
-
- static PLDHashOperator
- Enumerate(const uint64_t& aKey, FullIndexMetadata* aValue, void* aClosure)
- {
- MOZ_ASSERT(aKey);
- MOZ_ASSERT(aValue);
- MOZ_ASSERT(aClosure);
-
- auto* helper = static_cast(aClosure);
-
- MOZ_ASSERT(helper->mCurrentObjectStoreSpec);
-
- // XXX This should really be fallible...
- IndexMetadata* metadata =
- helper->mCurrentObjectStoreSpec->indexes().AppendElement();
- *metadata = aValue->mCommonMetadata;
-
- return PL_DHASH_NEXT;
- }
- };
-
- Helper::CopyToSpec(mMetadata, aSpec);
+ }
}
-
#ifdef DEBUG
void
@@ -21075,110 +21028,6 @@ OpenDatabaseOp::AssertMetadataConsistency(const FullDatabaseMetadata* aMetadata)
{
AssertIsOnBackgroundThread();
- class MOZ_STACK_CLASS Helper final
- {
- const ObjectStoreTable& mOtherObjectStores;
- IndexTable* mCurrentOtherIndexTable;
-
- public:
- static void
- AssertConsistent(const ObjectStoreTable& aThisObjectStores,
- const ObjectStoreTable& aOtherObjectStores)
- {
- Helper helper(aOtherObjectStores);
- aThisObjectStores.EnumerateRead(Enumerate, &helper);
- }
-
- private:
- explicit Helper(const ObjectStoreTable& aOtherObjectStores)
- : mOtherObjectStores(aOtherObjectStores)
- , mCurrentOtherIndexTable(nullptr)
- { }
-
- static PLDHashOperator
- Enumerate(const uint64_t& /* aKey */,
- FullObjectStoreMetadata* aThisObjectStore,
- void* aClosure)
- {
- MOZ_ASSERT(aThisObjectStore);
- MOZ_ASSERT(!aThisObjectStore->mDeleted);
- MOZ_ASSERT(aClosure);
-
- auto* helper = static_cast(aClosure);
-
- MOZ_ASSERT(!helper->mCurrentOtherIndexTable);
-
- auto* otherObjectStore =
- MetadataNameOrIdMatcher::Match(
- helper->mOtherObjectStores, aThisObjectStore->mCommonMetadata.id());
- MOZ_ASSERT(otherObjectStore);
-
- MOZ_ASSERT(aThisObjectStore != otherObjectStore);
-
- MOZ_ASSERT(aThisObjectStore->mCommonMetadata.id() ==
- otherObjectStore->mCommonMetadata.id());
- MOZ_ASSERT(aThisObjectStore->mCommonMetadata.name() ==
- otherObjectStore->mCommonMetadata.name());
- MOZ_ASSERT(aThisObjectStore->mCommonMetadata.autoIncrement() ==
- otherObjectStore->mCommonMetadata.autoIncrement());
- MOZ_ASSERT(aThisObjectStore->mCommonMetadata.keyPath() ==
- otherObjectStore->mCommonMetadata.keyPath());
- // mNextAutoIncrementId and mComittedAutoIncrementId may be modified
- // concurrently with this OpenOp, so it is not possible to assert equality
- // here.
- MOZ_ASSERT(aThisObjectStore->mNextAutoIncrementId <=
- otherObjectStore->mNextAutoIncrementId);
- MOZ_ASSERT(aThisObjectStore->mComittedAutoIncrementId <=
- otherObjectStore->mComittedAutoIncrementId);
- MOZ_ASSERT(!otherObjectStore->mDeleted);
-
- MOZ_ASSERT(aThisObjectStore->mIndexes.Count() ==
- otherObjectStore->mIndexes.Count());
-
- AutoRestore ar(helper->mCurrentOtherIndexTable);
- helper->mCurrentOtherIndexTable = &otherObjectStore->mIndexes;
-
- aThisObjectStore->mIndexes.EnumerateRead(Enumerate, helper);
-
- return PL_DHASH_NEXT;
- }
-
- static PLDHashOperator
- Enumerate(const uint64_t& /* aKey */,
- FullIndexMetadata* aThisIndex,
- void* aClosure)
- {
- MOZ_ASSERT(aThisIndex);
- MOZ_ASSERT(!aThisIndex->mDeleted);
- MOZ_ASSERT(aClosure);
-
- auto* helper = static_cast(aClosure);
-
- MOZ_ASSERT(helper->mCurrentOtherIndexTable);
-
- auto* otherIndex =
- MetadataNameOrIdMatcher::Match(
- *helper->mCurrentOtherIndexTable, aThisIndex->mCommonMetadata.id());
- MOZ_ASSERT(otherIndex);
-
- MOZ_ASSERT(aThisIndex != otherIndex);
-
- MOZ_ASSERT(aThisIndex->mCommonMetadata.id() ==
- otherIndex->mCommonMetadata.id());
- MOZ_ASSERT(aThisIndex->mCommonMetadata.name() ==
- otherIndex->mCommonMetadata.name());
- MOZ_ASSERT(aThisIndex->mCommonMetadata.keyPath() ==
- otherIndex->mCommonMetadata.keyPath());
- MOZ_ASSERT(aThisIndex->mCommonMetadata.unique() ==
- otherIndex->mCommonMetadata.unique());
- MOZ_ASSERT(aThisIndex->mCommonMetadata.multiEntry() ==
- otherIndex->mCommonMetadata.multiEntry());
- MOZ_ASSERT(!otherIndex->mDeleted);
-
- return PL_DHASH_NEXT;
- }
- };
-
const FullDatabaseMetadata* thisDB = mMetadata;
const FullDatabaseMetadata* otherDB = aMetadata;
@@ -21203,7 +21052,67 @@ OpenDatabaseOp::AssertMetadataConsistency(const FullDatabaseMetadata* aMetadata)
MOZ_ASSERT(thisDB->mObjectStores.Count() == otherDB->mObjectStores.Count());
- Helper::AssertConsistent(thisDB->mObjectStores, otherDB->mObjectStores);
+ for (auto objectStoreIter = thisDB->mObjectStores.ConstIter();
+ !objectStoreIter.Done();
+ objectStoreIter.Next()) {
+ FullObjectStoreMetadata* thisObjectStore = objectStoreIter.UserData();
+ MOZ_ASSERT(thisObjectStore);
+ MOZ_ASSERT(!thisObjectStore->mDeleted);
+
+ auto* otherObjectStore =
+ MetadataNameOrIdMatcher::Match(
+ otherDB->mObjectStores, thisObjectStore->mCommonMetadata.id());
+ MOZ_ASSERT(otherObjectStore);
+
+ MOZ_ASSERT(thisObjectStore != otherObjectStore);
+
+ MOZ_ASSERT(thisObjectStore->mCommonMetadata.id() ==
+ otherObjectStore->mCommonMetadata.id());
+ MOZ_ASSERT(thisObjectStore->mCommonMetadata.name() ==
+ otherObjectStore->mCommonMetadata.name());
+ MOZ_ASSERT(thisObjectStore->mCommonMetadata.autoIncrement() ==
+ otherObjectStore->mCommonMetadata.autoIncrement());
+ MOZ_ASSERT(thisObjectStore->mCommonMetadata.keyPath() ==
+ otherObjectStore->mCommonMetadata.keyPath());
+ // mNextAutoIncrementId and mComittedAutoIncrementId may be modified
+ // concurrently with this OpenOp, so it is not possible to assert equality
+ // here.
+ MOZ_ASSERT(thisObjectStore->mNextAutoIncrementId <=
+ otherObjectStore->mNextAutoIncrementId);
+ MOZ_ASSERT(thisObjectStore->mComittedAutoIncrementId <=
+ otherObjectStore->mComittedAutoIncrementId);
+ MOZ_ASSERT(!otherObjectStore->mDeleted);
+
+ MOZ_ASSERT(thisObjectStore->mIndexes.Count() ==
+ otherObjectStore->mIndexes.Count());
+
+ for (auto indexIter = thisObjectStore->mIndexes.Iter();
+ !indexIter.Done();
+ indexIter.Next()) {
+ FullIndexMetadata* thisIndex = indexIter.UserData();
+ MOZ_ASSERT(thisIndex);
+ MOZ_ASSERT(!thisIndex->mDeleted);
+
+ auto* otherIndex =
+ MetadataNameOrIdMatcher::
+ Match(otherObjectStore->mIndexes, thisIndex->mCommonMetadata.id());
+ MOZ_ASSERT(otherIndex);
+
+ MOZ_ASSERT(thisIndex != otherIndex);
+
+ MOZ_ASSERT(thisIndex->mCommonMetadata.id() ==
+ otherIndex->mCommonMetadata.id());
+ MOZ_ASSERT(thisIndex->mCommonMetadata.name() ==
+ otherIndex->mCommonMetadata.name());
+ MOZ_ASSERT(thisIndex->mCommonMetadata.keyPath() ==
+ otherIndex->mCommonMetadata.keyPath());
+ MOZ_ASSERT(thisIndex->mCommonMetadata.unique() ==
+ otherIndex->mCommonMetadata.unique());
+ MOZ_ASSERT(thisIndex->mCommonMetadata.multiEntry() ==
+ otherIndex->mCommonMetadata.multiEntry());
+ MOZ_ASSERT(!otherIndex->mDeleted);
+ }
+ }
}
#endif // DEBUG
diff --git a/dom/indexedDB/IndexedDatabaseManager.cpp b/dom/indexedDB/IndexedDatabaseManager.cpp
index 0a19a83046d..7bc91fc89ca 100644
--- a/dom/indexedDB/IndexedDatabaseManager.cpp
+++ b/dom/indexedDB/IndexedDatabaseManager.cpp
@@ -262,7 +262,7 @@ IndexedDatabaseManager::~IndexedDatabaseManager()
bool IndexedDatabaseManager::sIsMainProcess = false;
bool IndexedDatabaseManager::sFullSynchronousMode = false;
-PRLogModuleInfo* IndexedDatabaseManager::sLoggingModule;
+mozilla::LazyLogModule IndexedDatabaseManager::sLoggingModule("IndexedDB");
Atomic
IndexedDatabaseManager::sLoggingMode(
@@ -284,10 +284,6 @@ IndexedDatabaseManager::GetOrCreate()
if (!gDBManager) {
sIsMainProcess = XRE_IsParentProcess();
- if (!sLoggingModule) {
- sLoggingModule = PR_NewLogModule("IndexedDB");
- }
-
if (sIsMainProcess && Preferences::GetBool("disk_space_watcher.enabled", false)) {
// See if we're starting up in low disk space conditions.
nsCOMPtr watcher =
@@ -674,7 +670,7 @@ IndexedDatabaseManager::GetLoggingMode()
}
// static
-PRLogModuleInfo*
+mozilla::LogModule*
IndexedDatabaseManager::GetLoggingModule()
{
MOZ_ASSERT(gDBManager,
diff --git a/dom/indexedDB/IndexedDatabaseManager.h b/dom/indexedDB/IndexedDatabaseManager.h
index 4ad1191a65a..3bbb53d2010 100644
--- a/dom/indexedDB/IndexedDatabaseManager.h
+++ b/dom/indexedDB/IndexedDatabaseManager.h
@@ -19,7 +19,6 @@
#include "nsITimer.h"
class nsIEventTarget;
-struct PRLogModuleInfo;
namespace mozilla {
@@ -101,7 +100,7 @@ public:
}
#endif
- static PRLogModuleInfo*
+ static mozilla::LogModule*
GetLoggingModule()
#ifdef DEBUG
;
@@ -226,7 +225,7 @@ private:
static bool sIsMainProcess;
static bool sFullSynchronousMode;
- static PRLogModuleInfo* sLoggingModule;
+ static LazyLogModule sLoggingModule;
static Atomic sLoggingMode;
static mozilla::Atomic sLowDiskSpaceMode;
};
diff --git a/dom/indexedDB/ProfilerHelpers.h b/dom/indexedDB/ProfilerHelpers.h
index 70a720c02fa..00fc5b88b1e 100644
--- a/dom/indexedDB/ProfilerHelpers.h
+++ b/dom/indexedDB/ProfilerHelpers.h
@@ -279,7 +279,7 @@ LoggingHelper(bool aUseProfiler, const char* aFmt, ...)
IndexedDatabaseManager::Logging_Disabled);
MOZ_ASSERT(aFmt);
- PRLogModuleInfo* logModule = IndexedDatabaseManager::GetLoggingModule();
+ mozilla::LogModule* logModule = IndexedDatabaseManager::GetLoggingModule();
MOZ_ASSERT(logModule);
static const mozilla::LogLevel logLevel = LogLevel::Warning;
diff --git a/dom/ipc/ProcessPriorityManager.cpp b/dom/ipc/ProcessPriorityManager.cpp
index 5811be9234a..ea80c94088f 100644
--- a/dom/ipc/ProcessPriorityManager.cpp
+++ b/dom/ipc/ProcessPriorityManager.cpp
@@ -28,6 +28,10 @@
#include "nsComponentManagerUtils.h"
#include "nsCRT.h"
+using namespace mozilla;
+using namespace mozilla::dom;
+using namespace mozilla::hal;
+
#ifdef XP_WIN
#include
#define getpid _getpid
@@ -69,12 +73,10 @@
NameWithComma().get(), \
static_cast(ChildID()), Pid(), ##__VA_ARGS__)
#else
- static PRLogModuleInfo*
+ static LogModule*
GetPPMLog()
{
- static PRLogModuleInfo *sLog;
- if (!sLog)
- sLog = PR_NewLogModule("ProcessPriorityManager");
+ static LazyLogModule sLog("ProcessPriorityManager");
return sLog;
}
# define LOG(fmt, ...) \
@@ -87,10 +89,6 @@
static_cast(ChildID()), Pid(), ##__VA_ARGS__))
#endif
-using namespace mozilla;
-using namespace mozilla::dom;
-using namespace mozilla::hal;
-
namespace {
class ParticularProcessPriorityManager;
diff --git a/dom/ipc/ScreenManagerParent.cpp b/dom/ipc/ScreenManagerParent.cpp
index 747d938fcff..e204b2163c8 100644
--- a/dom/ipc/ScreenManagerParent.cpp
+++ b/dom/ipc/ScreenManagerParent.cpp
@@ -142,11 +142,9 @@ ScreenManagerParent::RecvScreenForBrowser(const TabId& aTabId,
nsCOMPtr widget = tabParent->GetWidget();
nsCOMPtr screen;
- if (widget) {
- if (widget->GetNativeData(NS_NATIVE_WINDOW)) {
- mScreenMgr->ScreenForNativeWidget(widget->GetNativeData(NS_NATIVE_WINDOW),
- getter_AddRefs(screen));
- }
+ if (widget && widget->GetNativeData(NS_NATIVE_WINDOW)) {
+ mScreenMgr->ScreenForNativeWidget(widget->GetNativeData(NS_NATIVE_WINDOW),
+ getter_AddRefs(screen));
} else {
nsresult rv = mScreenMgr->GetPrimaryScreen(getter_AddRefs(screen));
if (NS_WARN_IF(NS_FAILED(rv))) {
diff --git a/dom/media/MP3Demuxer.cpp b/dom/media/MP3Demuxer.cpp
index f7ce9ae7011..fc4a47862fb 100644
--- a/dom/media/MP3Demuxer.cpp
+++ b/dom/media/MP3Demuxer.cpp
@@ -161,7 +161,11 @@ MP3TrackDemuxer::DemuxSample() {
media::TimeUnit
MP3TrackDemuxer::SeekPosition() const {
- return Duration(mFrameIndex);
+ TimeUnit pos = Duration(mFrameIndex);
+ if (Duration() > TimeUnit()) {
+ pos = std::min(Duration(), pos);
+ }
+ return pos;
}
#endif
@@ -199,22 +203,22 @@ MP3TrackDemuxer::FastSeek(const TimeUnit& aTime) {
const auto& vbr = mParser.VBRInfo();
if (!aTime.ToMicroseconds()) {
// Quick seek to the beginning of the stream.
- mOffset = mFirstFrameOffset;
+ mFrameIndex = 0;
} else if (vbr.IsTOCPresent()) {
// Use TOC for more precise seeking.
const float durationFrac = static_cast(aTime.ToMicroseconds()) /
Duration().ToMicroseconds();
- mOffset = vbr.Offset(durationFrac);
+ mFrameIndex = FrameIndexFromOffset(vbr.Offset(durationFrac));
} else if (AverageFrameLength() > 0) {
- mOffset = mFirstFrameOffset + FrameIndexFromTime(aTime) *
- AverageFrameLength();
+ mFrameIndex = FrameIndexFromTime(aTime);
}
+ mOffset = OffsetFromFrameIndex(mFrameIndex);
+
if (mOffset > mFirstFrameOffset && StreamLength() > 0) {
mOffset = std::min(StreamLength() - 1, mOffset);
}
- mFrameIndex = FrameIndexFromOffset(mOffset);
mParser.EndFrameSession();
MP3LOG("FastSeek End TOC=%d avgFrameLen=%f mNumParsedFrames=%" PRIu64
@@ -240,6 +244,10 @@ MP3TrackDemuxer::ScanUntil(const TimeUnit& aTime) {
FastSeek(aTime);
}
+ if (Duration(mFrameIndex + 1) > aTime) {
+ return SeekPosition();
+ }
+
MediaByteRange nextRange = FindNextFrame();
while (SkipNextFrame(nextRange) && Duration(mFrameIndex + 1) < aTime) {
nextRange = FindNextFrame();
@@ -253,7 +261,7 @@ MP3TrackDemuxer::ScanUntil(const TimeUnit& aTime) {
" mFrameIndex=%" PRId64 " mOffset=%" PRIu64,
aTime, AverageFrameLength(), mNumParsedFrames, mFrameIndex, mOffset);
- return Duration(mFrameIndex);
+ return SeekPosition();
}
RefPtr
@@ -363,10 +371,26 @@ MP3TrackDemuxer::Duration(int64_t aNumFrames) const {
return TimeUnit::FromMicroseconds(aNumFrames * usPerFrame);
}
+static bool
+VerifyFrameConsistency(
+ const FrameParser::Frame& aFrame1, const FrameParser::Frame& aFrame2) {
+ const auto& h1 = aFrame1.Header();
+ const auto& h2 = aFrame2.Header();
+
+ return h1.IsValid() && h2.IsValid() &&
+ h1.Layer() == h2.Layer() &&
+ h1.SlotSize() == h2.SlotSize() &&
+ h1.SamplesPerFrame() == h2.SamplesPerFrame() &&
+ h1.Channels() == h2.Channels() &&
+ h1.SampleRate() == h2.SampleRate() &&
+ h1.RawVersion() == h2.RawVersion() &&
+ h1.RawProtection() == h2.RawProtection();
+}
+
MediaByteRange
MP3TrackDemuxer::FindNextFrame() {
- static const int BUFFER_SIZE = 4096;
- static const int MAX_SKIPPED_BYTES = 10 * BUFFER_SIZE;
+ static const int BUFFER_SIZE = 64;
+ static const int MAX_SKIPPED_BYTES = 1024 * BUFFER_SIZE;
MP3LOGV("FindNext() Begin mOffset=%" PRIu64 " mNumParsedFrames=%" PRIu64
" mFrameIndex=%" PRId64 " mTotalFrameLen=%" PRIu64
@@ -400,10 +424,20 @@ MP3TrackDemuxer::FindNextFrame() {
MOZ_ASSERT(foundFrame || bytesToSkip || !reader.Remaining());
reader.DiscardRemaining();
- // Advance mOffset by the amount of bytes read and if necessary,
- // skip an ID3v2 tag which stretches beyond the current buffer.
- NS_ENSURE_TRUE(mOffset + read + bytesToSkip > mOffset, MediaByteRange(0, 0));
- mOffset += read + bytesToSkip;
+ if (foundFrame && mParser.FirstFrame().Length() &&
+ !VerifyFrameConsistency(mParser.FirstFrame(), mParser.CurrentFrame())) {
+ // We've likely hit a false-positive, ignore it and proceed with the
+ // search for the next valid frame.
+ foundFrame = false;
+ mOffset = frameHeaderOffset + 1;
+ mParser.EndFrameSession();
+ } else {
+ // Advance mOffset by the amount of bytes read and if necessary,
+ // skip an ID3v2 tag which stretches beyond the current buffer.
+ NS_ENSURE_TRUE(mOffset + read + bytesToSkip > mOffset,
+ MediaByteRange(0, 0));
+ mOffset += read + bytesToSkip;
+ }
}
if (!foundFrame || !mParser.CurrentFrame().Length()) {
@@ -493,6 +527,22 @@ MP3TrackDemuxer::GetNextFrame(const MediaByteRange& aRange) {
return frame.forget();
}
+int64_t
+MP3TrackDemuxer::OffsetFromFrameIndex(int64_t aFrameIndex) const {
+ int64_t offset = 0;
+ const auto& vbr = mParser.VBRInfo();
+
+ if (vbr.NumBytes() && vbr.NumAudioFrames()) {
+ offset = mFirstFrameOffset + aFrameIndex * vbr.NumBytes().value() /
+ vbr.NumAudioFrames().value();
+ } else if (AverageFrameLength() > 0) {
+ offset = mFirstFrameOffset + aFrameIndex * AverageFrameLength();
+ }
+
+ MP3LOGV("OffsetFromFrameIndex(%" PRId64 ") -> %" PRId64, aFrameIndex, offset);
+ return std::max(mFirstFrameOffset, offset);
+}
+
int64_t
MP3TrackDemuxer::FrameIndexFromOffset(int64_t aOffset) const {
int64_t frameIndex = 0;
@@ -846,7 +896,7 @@ FrameParser::FrameHeader::IsValid(int aPos) const {
if (aPos == frame_header::SYNC2_VERSION_LAYER_PROTECTION) {
return Sync2() == 7 &&
RawVersion() != 1 &&
- RawLayer() != 0;
+ Layer() == 3;
}
if (aPos == frame_header::BITRATE_SAMPLERATE_PADDING_PRIVATE) {
return RawBitrate() != 0xF && RawBitrate() != 0 &&
@@ -1135,6 +1185,9 @@ ID3Parser::ID3Header::Flags() const {
uint32_t
ID3Parser::ID3Header::Size() const {
+ if (!IsValid()) {
+ return 0;
+ }
return mSize;
}
diff --git a/dom/media/MP3Demuxer.h b/dom/media/MP3Demuxer.h
index fdcffbf7af8..0dc664c89b3 100644
--- a/dom/media/MP3Demuxer.h
+++ b/dom/media/MP3Demuxer.h
@@ -411,10 +411,13 @@ private:
// Updates post-read meta data.
void UpdateState(const MediaByteRange& aRange);
- // Returns the frame index for the given offset.
+ // Returns the estimated offset for the given frame index.
+ int64_t OffsetFromFrameIndex(int64_t aFrameIndex) const;
+
+ // Returns the estimated frame index for the given offset.
int64_t FrameIndexFromOffset(int64_t aOffset) const;
- // Returns the frame index for the given time.
+ // Returns the estimated frame index for the given time.
int64_t FrameIndexFromTime(const media::TimeUnit& aTime) const;
// Reads aSize bytes into aBuffer from the source starting at aOffset.
diff --git a/dom/media/MediaDecoderOwner.h b/dom/media/MediaDecoderOwner.h
index d1cf5608123..dda03b7578a 100644
--- a/dom/media/MediaDecoderOwner.h
+++ b/dom/media/MediaDecoderOwner.h
@@ -117,10 +117,10 @@ public:
};
// Check if the decoder owner is active.
- virtual bool IsActive() = 0;
+ virtual bool IsActive() const = 0;
// Check if the decoder owner is hidden.
- virtual bool IsHidden() = 0;
+ virtual bool IsHidden() const = 0;
// Called by the media decoder and the video frame to get the
// ImageContainer containing the video data.
diff --git a/dom/media/eme/EMEUtils.cpp b/dom/media/eme/EMEUtils.cpp
index b89879a6f8e..c75984387d7 100644
--- a/dom/media/eme/EMEUtils.cpp
+++ b/dom/media/eme/EMEUtils.cpp
@@ -5,6 +5,7 @@
* You can obtain one at http://mozilla.org/MPL/2.0/. */
#include "mozilla/EMEUtils.h"
+#include "mozilla/dom/UnionTypes.h"
namespace mozilla {
@@ -112,4 +113,32 @@ ConstructKeySystem(const nsAString& aKeySystem,
aOutKeySystem.Append(aCDMVersion);
}
+ArrayData
+GetArrayBufferViewOrArrayBufferData(const dom::ArrayBufferViewOrArrayBuffer& aBufferOrView)
+{
+ MOZ_ASSERT(aBufferOrView.IsArrayBuffer() || aBufferOrView.IsArrayBufferView());
+ if (aBufferOrView.IsArrayBuffer()) {
+ const dom::ArrayBuffer& buffer = aBufferOrView.GetAsArrayBuffer();
+ buffer.ComputeLengthAndData();
+ return ArrayData(buffer.Data(), buffer.Length());
+ } else if (aBufferOrView.IsArrayBufferView()) {
+ const dom::ArrayBufferView& bufferview = aBufferOrView.GetAsArrayBufferView();
+ bufferview.ComputeLengthAndData();
+ return ArrayData(bufferview.Data(), bufferview.Length());
+ }
+ return ArrayData(nullptr, 0);
+}
+
+void
+CopyArrayBufferViewOrArrayBufferData(const dom::ArrayBufferViewOrArrayBuffer& aBufferOrView,
+ nsTArray& aOutData)
+{
+ ArrayData data = GetArrayBufferViewOrArrayBufferData(aBufferOrView);
+ aOutData.Clear();
+ if (!data.IsValid()) {
+ return;
+ }
+ aOutData.AppendElements(data.mData, data.mLength);
+}
+
} // namespace mozilla
diff --git a/dom/media/eme/EMEUtils.h b/dom/media/eme/EMEUtils.h
index 6768213b5d9..64a01f5499b 100644
--- a/dom/media/eme/EMEUtils.h
+++ b/dom/media/eme/EMEUtils.h
@@ -9,9 +9,14 @@
#include "mozilla/Logging.h"
#include "nsString.h"
+#include "nsTArray.h"
namespace mozilla {
+namespace dom {
+class ArrayBufferViewOrArrayBuffer;
+}
+
#ifndef EME_LOG
LogModule* GetEMELog();
#define EME_LOG(...) MOZ_LOG(GetEMELog(), mozilla::LogLevel::Debug, (__VA_ARGS__))
@@ -56,6 +61,49 @@ ConstructKeySystem(const nsAString& aKeySystem,
const nsAString& aCDMVersion,
nsAString& aOutKeySystem);
+// Helper function to extract a copy of data coming in from JS in an
+// (ArrayBuffer or ArrayBufferView) IDL typed function argument.
+//
+// Only call this on a properly initialized ArrayBufferViewOrArrayBuffer.
+void
+CopyArrayBufferViewOrArrayBufferData(const dom::ArrayBufferViewOrArrayBuffer& aBufferOrView,
+ nsTArray& aOutData);
+
+struct ArrayData {
+ explicit ArrayData(const uint8_t* aData, size_t aLength)
+ : mData(aData)
+ , mLength(aLength)
+ {
+ }
+ const uint8_t* mData;
+ const size_t mLength;
+ bool IsValid() const {
+ return mData != nullptr && mLength != 0;
+ }
+ bool operator== (const nsTArray& aOther) const {
+ return mLength == aOther.Length() &&
+ memcmp(mData, aOther.Elements(), mLength) == 0;
+ }
+};
+
+// Helper function to extract data coming in from JS in an
+// (ArrayBuffer or ArrayBufferView) IDL typed function argument.
+//
+// Be *very* careful with this!
+//
+// Only use returned ArrayData inside the lifetime of the
+// ArrayBufferViewOrArrayBuffer; the ArrayData struct does not contain
+// a copy of the data!
+//
+// And do *not* call out to anything that could call into JavaScript,
+// while the ArrayData is live, as then all bets about the data not changing
+// are off! No calls into JS, no calls into JS-implemented WebIDL or XPIDL,
+// nothing. Beware!
+//
+// Only call this on a properly initialized ArrayBufferViewOrArrayBuffer.
+ArrayData
+GetArrayBufferViewOrArrayBufferData(const dom::ArrayBufferViewOrArrayBuffer& aBufferOrView);
+
} // namespace mozilla
#endif // EME_LOG_H_
diff --git a/dom/media/eme/MediaKeySession.cpp b/dom/media/eme/MediaKeySession.cpp
index 8c31c00c726..4b852dbc994 100644
--- a/dom/media/eme/MediaKeySession.cpp
+++ b/dom/media/eme/MediaKeySession.cpp
@@ -54,7 +54,7 @@ MediaKeySession::MediaKeySession(JSContext* aCx,
, mToken(sMediaKeySessionNum++)
, mIsClosed(false)
, mUninitialized(true)
- , mKeyStatusMap(new MediaKeyStatusMap(aCx, aParent, aRv))
+ , mKeyStatusMap(new MediaKeyStatusMap(aParent))
, mExpiration(JS::GenericNaN())
{
EME_LOG("MediaKeySession[%p,''] session Id set", this);
@@ -124,7 +124,6 @@ MediaKeySession::Closed() const
return mClosed;
}
-
void
MediaKeySession::UpdateKeyStatusMap()
{
@@ -175,19 +174,26 @@ MediaKeySession::GenerateRequest(const nsAString& aInitDataType,
EME_LOG("MediaKeySession[%p,'%s'] GenerateRequest() failed, uninitialized",
this, NS_ConvertUTF16toUTF8(mSessionId).get());
promise->MaybeReject(NS_ERROR_DOM_INVALID_ACCESS_ERR,
- NS_LITERAL_CSTRING("Session is already initialized in MediaKeySession.generateRequest()"));
+ NS_LITERAL_CSTRING("Session is already initialized in MediaKeySession.generateRequest()"));
return promise.forget();
}
mUninitialized = false;
+ if (aInitDataType.IsEmpty()) {
+ promise->MaybeReject(NS_ERROR_DOM_TYPE_ERR,
+ NS_LITERAL_CSTRING("Empty initDataType passed to MediaKeySession.generateRequest()"));
+ EME_LOG("MediaKeySession[%p,'%s'] GenerateRequest() failed, empty initDataType",
+ this, NS_ConvertUTF16toUTF8(mSessionId).get());
+ return promise.forget();
+ }
+
nsTArray data;
- if (aInitDataType.IsEmpty() ||
- !CopyArrayBufferViewOrArrayBufferData(aInitData, data)) {
- promise->MaybeReject(NS_ERROR_DOM_INVALID_ACCESS_ERR,
- NS_LITERAL_CSTRING("Bad arguments to MediaKeySession.generateRequest()"));
- EME_LOG("MediaKeySession[%p,'%s'] GenerateRequest() failed, "
- "invalid initData or initDataType",
+ CopyArrayBufferViewOrArrayBufferData(aInitData, data);
+ if (data.IsEmpty()) {
+ promise->MaybeReject(NS_ERROR_DOM_TYPE_ERR,
+ NS_LITERAL_CSTRING("Empty initData passed to MediaKeySession.generateRequest()"));
+ EME_LOG("MediaKeySession[%p,'%s'] GenerateRequest() failed, empty initData",
this, NS_ConvertUTF16toUTF8(mSessionId).get());
return promise.forget();
}
@@ -282,10 +288,11 @@ MediaKeySession::Update(const ArrayBufferViewOrArrayBuffer& aResponse, ErrorResu
this, NS_ConvertUTF16toUTF8(mSessionId).get());
return promise.forget();
}
- if (!CopyArrayBufferViewOrArrayBufferData(aResponse, data)) {
- promise->MaybeReject(NS_ERROR_DOM_INVALID_ACCESS_ERR,
- NS_LITERAL_CSTRING("Invalid response buffer"));
- EME_LOG("MediaKeySession[%p,'%s'] Update() failed, invalid response buffer",
+ CopyArrayBufferViewOrArrayBufferData(aResponse, data);
+ if (data.IsEmpty()) {
+ promise->MaybeReject(NS_ERROR_DOM_TYPE_ERR,
+ NS_LITERAL_CSTRING("Empty response buffer passed to MediaKeySession.update()"));
+ EME_LOG("MediaKeySession[%p,'%s'] Update() failed, empty response buffer",
this, NS_ConvertUTF16toUTF8(mSessionId).get());
return promise.forget();
}
diff --git a/dom/media/eme/MediaKeyStatusMap.cpp b/dom/media/eme/MediaKeyStatusMap.cpp
index 7084176b48e..694475e1c76 100644
--- a/dom/media/eme/MediaKeyStatusMap.cpp
+++ b/dom/media/eme/MediaKeyStatusMap.cpp
@@ -8,53 +8,26 @@
#include "nsPIDOMWindow.h"
#include "mozilla/dom/UnionTypes.h"
#include "mozilla/dom/ToJSValue.h"
+#include "mozilla/EMEUtils.h"
namespace mozilla {
namespace dom {
NS_IMPL_CYCLE_COLLECTING_ADDREF(MediaKeyStatusMap)
NS_IMPL_CYCLE_COLLECTING_RELEASE(MediaKeyStatusMap)
-
NS_INTERFACE_MAP_BEGIN_CYCLE_COLLECTION(MediaKeyStatusMap)
NS_WRAPPERCACHE_INTERFACE_MAP_ENTRY
NS_INTERFACE_MAP_ENTRY(nsISupports)
NS_INTERFACE_MAP_END
+NS_IMPL_CYCLE_COLLECTION_WRAPPERCACHE(MediaKeyStatusMap, mParent)
-NS_IMPL_CYCLE_COLLECTION_CLASS(MediaKeyStatusMap)
-
-NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(MediaKeyStatusMap)
- NS_IMPL_CYCLE_COLLECTION_UNLINK(mParent)
- NS_IMPL_CYCLE_COLLECTION_UNLINK_PRESERVED_WRAPPER
- tmp->mMap = nullptr;
-NS_IMPL_CYCLE_COLLECTION_UNLINK_END
-
-NS_IMPL_CYCLE_COLLECTION_TRAVERSE_BEGIN(MediaKeyStatusMap)
- NS_IMPL_CYCLE_COLLECTION_TRAVERSE(mParent)
- NS_IMPL_CYCLE_COLLECTION_TRAVERSE_SCRIPT_OBJECTS
-NS_IMPL_CYCLE_COLLECTION_TRAVERSE_END
-
-NS_IMPL_CYCLE_COLLECTION_TRACE_BEGIN(MediaKeyStatusMap)
- NS_IMPL_CYCLE_COLLECTION_TRACE_PRESERVED_WRAPPER
- NS_IMPL_CYCLE_COLLECTION_TRACE_JS_MEMBER_CALLBACK(mMap)
-NS_IMPL_CYCLE_COLLECTION_TRACE_END
-
-MediaKeyStatusMap::MediaKeyStatusMap(JSContext* aCx,
- nsPIDOMWindow* aParent,
- ErrorResult& aRv)
+MediaKeyStatusMap::MediaKeyStatusMap(nsPIDOMWindow* aParent)
: mParent(aParent)
- , mUpdateError(NS_OK)
{
- mMap = JS::NewMapObject(aCx);
- if (NS_WARN_IF(!mMap)) {
- aRv.Throw(NS_ERROR_OUT_OF_MEMORY);
- }
-
- mozilla::HoldJSObjects(this);
}
MediaKeyStatusMap::~MediaKeyStatusMap()
{
- mozilla::DropJSObjects(this);
}
JSObject*
@@ -70,110 +43,63 @@ MediaKeyStatusMap::GetParentObject() const
}
MediaKeyStatus
-MediaKeyStatusMap::Get(JSContext* aCx,
- const ArrayBufferViewOrArrayBuffer& aKey,
- ErrorResult& aRv) const
+MediaKeyStatusMap::Get(const ArrayBufferViewOrArrayBuffer& aKey) const
{
- if (NS_FAILED(mUpdateError)) {
- aRv.Throw(mUpdateError);
+ ArrayData keyId = GetArrayBufferViewOrArrayBufferData(aKey);
+ if (!keyId.IsValid()) {
return MediaKeyStatus::Internal_error;
}
- JS::Rooted map(aCx, mMap);
- JS::Rooted key(aCx);
- JS::Rooted val(aCx);
-
- if (!aKey.ToJSVal(aCx, map, &key) ||
- !JS::MapGet(aCx, map, key, &val)) {
- aRv.Throw(NS_ERROR_FAILURE);
- return MediaKeyStatus::Internal_error;
+ for (const KeyStatus& status : mStatuses) {
+ if (keyId == status.mKeyId) {
+ return status.mStatus;
+ }
}
- bool ok;
- int index = FindEnumStringIndex(
- aCx, val, MediaKeyStatusValues::strings,
- "MediaKeyStatus", "Invalid MediaKeyStatus value", &ok);
-
- return ok ? static_cast(index) :
- MediaKeyStatus::Internal_error;
+ return MediaKeyStatus::Internal_error;
}
bool
-MediaKeyStatusMap::Has(JSContext* aCx,
- const ArrayBufferViewOrArrayBuffer& aKey,
- ErrorResult& aRv) const
+MediaKeyStatusMap::Has(const ArrayBufferViewOrArrayBuffer& aKey) const
{
- if (NS_FAILED(mUpdateError)) {
- aRv.Throw(mUpdateError);
+ ArrayData keyId = GetArrayBufferViewOrArrayBufferData(aKey);
+ if (!keyId.IsValid()) {
return false;
}
- JS::Rooted map(aCx, mMap);
- JS::Rooted key(aCx);
- bool result = false;
-
- if (!aKey.ToJSVal(aCx, map, &key) ||
- !JS::MapHas(aCx, map, key, &result)) {
- aRv.Throw(NS_ERROR_FAILURE);
+ for (const KeyStatus& status : mStatuses) {
+ if (keyId == status.mKeyId) {
+ return true;
+ }
}
- return result;
-}
-
-template
-static void CallMapMethod(JSContext* aCx,
- const JS::Heap& aMap,
- JS::MutableHandle aResult,
- ErrorResult& aRv,
- nsresult aUpdateError)
-{
- if (NS_FAILED(aUpdateError)) {
- aRv.Throw(aUpdateError);
- return;
- }
-
- JS::Rooted map(aCx, aMap);
- JS::Rooted result(aCx);
- if (!Method(aCx, map, &result)) {
- aRv.Throw(NS_ERROR_FAILURE);
- return;
- }
-
- aResult.set(&result.toObject());
-}
-
-void
-MediaKeyStatusMap::Keys(JSContext* aCx,
- JS::MutableHandle aResult,
- ErrorResult& aRv) const
-{
- CallMapMethod(aCx, mMap, aResult, aRv, mUpdateError);
-}
-
-void
-MediaKeyStatusMap::Values(JSContext* aCx,
- JS::MutableHandle aResult,
- ErrorResult& aRv) const
-{
- CallMapMethod(aCx, mMap, aResult, aRv, mUpdateError);
-}
-void
-MediaKeyStatusMap::Entries(JSContext* aCx,
- JS::MutableHandle aResult,
- ErrorResult& aRv) const
-{
- CallMapMethod(aCx, mMap, aResult, aRv, mUpdateError);
+ return false;
}
uint32_t
-MediaKeyStatusMap::GetSize(JSContext* aCx, ErrorResult& aRv) const
+MediaKeyStatusMap::GetIterableLength() const
{
- if (NS_FAILED(mUpdateError)) {
- aRv.Throw(mUpdateError);
- return 0;
- }
- JS::Rooted map(aCx, mMap);
- return JS::MapSize(aCx, map);
+ return mStatuses.Length();
+}
+
+TypedArrayCreator
+MediaKeyStatusMap::GetKeyAtIndex(uint32_t aIndex) const
+{
+ MOZ_ASSERT(aIndex < GetIterableLength());
+ return TypedArrayCreator(mStatuses[aIndex].mKeyId);
+}
+
+MediaKeyStatus
+MediaKeyStatusMap::GetValueAtIndex(uint32_t aIndex) const
+{
+ MOZ_ASSERT(aIndex < GetIterableLength());
+ return mStatuses[aIndex].mStatus;
+}
+
+uint32_t
+MediaKeyStatusMap::Size() const
+{
+ return mStatuses.Length();
}
static MediaKeyStatus
@@ -188,58 +114,13 @@ ToMediaKeyStatus(GMPMediaKeyStatus aStatus) {
}
}
-static bool
-ToJSString(JSContext* aCx, GMPMediaKeyStatus aStatus,
- JS::MutableHandle aResult)
-{
- auto val = uint32_t(ToMediaKeyStatus(aStatus));
- MOZ_ASSERT(val < ArrayLength(MediaKeyStatusValues::strings));
- JSString* str = JS_NewStringCopyN(aCx,
- MediaKeyStatusValues::strings[val].value,
- MediaKeyStatusValues::strings[val].length);
- if (!str) {
- return false;
- }
- aResult.setString(str);
- return true;
-}
-
-nsresult
-MediaKeyStatusMap::UpdateInternal(const nsTArray& keys)
-{
- AutoJSAPI jsapi;
- if (NS_WARN_IF(!jsapi.Init(mParent))) {
- return NS_ERROR_FAILURE;
- }
-
- jsapi.TakeOwnershipOfErrorReporting();
- JSContext* cx = jsapi.cx();
- JS::Rooted map(cx, mMap);
- if (!JS::MapClear(cx, map)) {
- return NS_ERROR_FAILURE;
- }
-
- for (size_t i = 0; i < keys.Length(); i++) {
- const auto& ks = keys[i];
- JS::Rooted key(cx);
- JS::Rooted val(cx);
- if (!ToJSValue(cx, TypedArrayCreator(ks.mId), &key) ||
- !ToJSString(cx, ks.mStatus, &val) ||
- !JS::MapSet(cx, map, key, val)) {
- return NS_ERROR_OUT_OF_MEMORY;
- }
- }
-
- return NS_OK;
-}
-
void
-MediaKeyStatusMap::Update(const nsTArray& keys)
+MediaKeyStatusMap::Update(const nsTArray& aKeys)
{
- // Since we can't leave the map in a partial update state, we need
- // to remember the error and throw it next time the interface methods
- // are called.
- mUpdateError = UpdateInternal(keys);
+ mStatuses.Clear();
+ for (const auto& key : aKeys) {
+ mStatuses.InsertElementSorted(KeyStatus(key.mId, ToMediaKeyStatus(key.mStatus)));
+ }
}
} // namespace dom
diff --git a/dom/media/eme/MediaKeyStatusMap.h b/dom/media/eme/MediaKeyStatusMap.h
index b82b5958430..7355563be0f 100644
--- a/dom/media/eme/MediaKeyStatusMap.h
+++ b/dom/media/eme/MediaKeyStatusMap.h
@@ -23,6 +23,9 @@ namespace dom {
class ArrayBufferViewOrArrayBuffer;
+// The MediaKeyStatusMap WebIDL interface; maps a keyId to its status.
+// Note that the underlying "map" is stored in an array, since we assume
+// that a MediaKeySession won't have many key statuses to report.
class MediaKeyStatusMap final : public nsISupports,
public nsWrapperCache
{
@@ -31,9 +34,7 @@ public:
NS_DECL_CYCLE_COLLECTION_SCRIPT_HOLDER_CLASS(MediaKeyStatusMap)
public:
- explicit MediaKeyStatusMap(JSContext* aCx,
- nsPIDOMWindow* aParent,
- ErrorResult& aRv);
+ explicit MediaKeyStatusMap(nsPIDOMWindow* aParent);
protected:
~MediaKeyStatusMap();
@@ -43,36 +44,48 @@ public:
virtual JSObject* WrapObject(JSContext* aCx, JS::Handle aGivenProto) override;
- MediaKeyStatus Get(JSContext* aCx,
- const ArrayBufferViewOrArrayBuffer& aKey,
- ErrorResult& aRv) const;
+ MediaKeyStatus Get(const ArrayBufferViewOrArrayBuffer& aKey) const;
+ bool Has(const ArrayBufferViewOrArrayBuffer& aKey) const;
+ uint32_t Size() const;
- bool Has(JSContext* aCx,
- const ArrayBufferViewOrArrayBuffer& aKey,
- ErrorResult& aRv) const;
-
- void Keys(JSContext* aCx,
- JS::MutableHandle aResult,
- ErrorResult& aRv) const;
-
- void Values(JSContext* aCx,
- JS::MutableHandle aResult,
- ErrorResult& aRv) const;
-
- void Entries(JSContext* aCx,
- JS::MutableHandle aResult,
- ErrorResult& aRv) const;
-
- uint32_t GetSize(JSContext* aCx, ErrorResult& aRv) const;
+ uint32_t GetIterableLength() const;
+ TypedArrayCreator GetKeyAtIndex(uint32_t aIndex) const;
+ MediaKeyStatus GetValueAtIndex(uint32_t aIndex) const;
void Update(const nsTArray& keys);
private:
- nsresult UpdateInternal(const nsTArray& keys);
nsCOMPtr mParent;
- JS::Heap mMap;
- nsresult mUpdateError;
+
+ struct KeyStatus {
+ KeyStatus(const nsTArray& aKeyId,
+ MediaKeyStatus aStatus)
+ : mKeyId(aKeyId)
+ , mStatus(aStatus)
+ {
+ }
+ bool operator== (const KeyStatus& aOther) const {
+ return aOther.mKeyId == mKeyId;
+ }
+ bool operator<(const KeyStatus& aOther) const {
+ // Copy chromium and compare keys' bytes.
+ // Update once https://github.com/w3c/encrypted-media/issues/69
+ // is resolved.
+ const nsTArray& other = aOther.mKeyId;
+ const nsTArray& self = mKeyId;
+ size_t length = std::min(other.Length(), self.Length());
+ int cmp = memcmp(self.Elements(), other.Elements(), length);
+ if (cmp != 0) {
+ return cmp < 0;
+ }
+ return self.Length() <= other.Length();
+ }
+ nsTArray mKeyId;
+ MediaKeyStatus mStatus;
+ };
+
+ nsTArray mStatuses;
};
} // namespace dom
diff --git a/dom/media/eme/MediaKeys.cpp b/dom/media/eme/MediaKeys.cpp
index 6663dfcb3a9..85c1b5be154 100644
--- a/dom/media/eme/MediaKeys.cpp
+++ b/dom/media/eme/MediaKeys.cpp
@@ -145,9 +145,10 @@ MediaKeys::SetServerCertificate(const ArrayBufferViewOrArrayBuffer& aCert, Error
}
nsTArray data;
- if (!CopyArrayBufferViewOrArrayBufferData(aCert, data)) {
- promise->MaybeReject(NS_ERROR_DOM_INVALID_ACCESS_ERR,
- NS_LITERAL_CSTRING("Invalid argument to MediaKeys.setServerCertificate()"));
+ CopyArrayBufferViewOrArrayBufferData(aCert, data);
+ if (data.IsEmpty()) {
+ promise->MaybeReject(NS_ERROR_DOM_TYPE_ERR,
+ NS_LITERAL_CSTRING("Empty certificate passed to MediaKeys.setServerCertificate()"));
return promise.forget();
}
@@ -502,23 +503,5 @@ MediaKeys::Bind(HTMLMediaElement* aElement)
return NS_OK;
}
-bool
-CopyArrayBufferViewOrArrayBufferData(const ArrayBufferViewOrArrayBuffer& aBufferOrView,
- nsTArray& aOutData)
-{
- if (aBufferOrView.IsArrayBuffer()) {
- const ArrayBuffer& buffer = aBufferOrView.GetAsArrayBuffer();
- buffer.ComputeLengthAndData();
- aOutData.AppendElements(buffer.Data(), buffer.Length());
- } else if (aBufferOrView.IsArrayBufferView()) {
- const ArrayBufferView& bufferview = aBufferOrView.GetAsArrayBufferView();
- bufferview.ComputeLengthAndData();
- aOutData.AppendElements(bufferview.Data(), bufferview.Length());
- } else {
- return false;
- }
- return true;
-}
-
} // namespace dom
} // namespace mozilla
diff --git a/dom/media/eme/MediaKeys.h b/dom/media/eme/MediaKeys.h
index fe9065421bf..07e68681abd 100644
--- a/dom/media/eme/MediaKeys.h
+++ b/dom/media/eme/MediaKeys.h
@@ -35,12 +35,6 @@ typedef nsRefPtrHashtable PromiseHashMap;
typedef nsRefPtrHashtable PendingKeySessionsHashMap;
typedef uint32_t PromiseId;
-// Helper function to extract data coming in from JS in an
-// (ArrayBuffer or ArrayBufferView) IDL typed function argument.
-bool
-CopyArrayBufferViewOrArrayBufferData(const ArrayBufferViewOrArrayBuffer& aBufferOrView,
- nsTArray& aOutData);
-
// This class is used on the main thread only.
// Note: it's addref/release is not (and can't be) thread safe!
class MediaKeys final : public nsISupports,
diff --git a/dom/media/gmp/GMPLoader.cpp b/dom/media/gmp/GMPLoader.cpp
index 1ebb1d470d1..99282ba2363 100644
--- a/dom/media/gmp/GMPLoader.cpp
+++ b/dom/media/gmp/GMPLoader.cpp
@@ -25,6 +25,7 @@
#ifdef XP_MACOSX
#include
#ifdef HASH_NODE_ID_WITH_DEVICE_ID
+#include
#include
#include
#endif
diff --git a/dom/media/gtest/MockMediaDecoderOwner.h b/dom/media/gtest/MockMediaDecoderOwner.h
index 9e28d13a0bd..148dfe879e6 100644
--- a/dom/media/gtest/MockMediaDecoderOwner.h
+++ b/dom/media/gtest/MockMediaDecoderOwner.h
@@ -36,8 +36,8 @@ public:
virtual void DispatchEncrypted(const nsTArray& aInitData,
const nsAString& aInitDataType) override {}
#endif // MOZ_EME
- virtual bool IsActive() override { return true; }
- virtual bool IsHidden() override { return false; }
+ virtual bool IsActive() const override { return true; }
+ virtual bool IsHidden() const override { return false; }
virtual void DownloadSuspended() override {}
virtual void DownloadResumed(bool aForceNetworkLoading) override {}
virtual void NotifySuspendedByCache(bool aIsSuspended) override {}
diff --git a/dom/media/gtest/MockMediaResource.cpp b/dom/media/gtest/MockMediaResource.cpp
index 5635e0f5c7f..32a77c916de 100644
--- a/dom/media/gtest/MockMediaResource.cpp
+++ b/dom/media/gtest/MockMediaResource.cpp
@@ -61,6 +61,9 @@ MockMediaResource::ReadAt(int64_t aOffset, char* aBuffer, uint32_t aCount,
int64_t
MockMediaResource::GetLength()
{
+ if (mFileHandle == nullptr) {
+ return -1;
+ }
fseek(mFileHandle, 0, SEEK_END);
return ftell(mFileHandle);
}
diff --git a/dom/media/gtest/TestMP3Demuxer.cpp b/dom/media/gtest/TestMP3Demuxer.cpp
index d4660d76c0d..eea8d9b45f1 100644
--- a/dom/media/gtest/TestMP3Demuxer.cpp
+++ b/dom/media/gtest/TestMP3Demuxer.cpp
@@ -12,6 +12,8 @@
using namespace mozilla;
using namespace mozilla::mp3;
+using media::TimeUnit;
+
// Regular MP3 file mock resource.
class MockMP3MediaResource : public MockMediaResource {
@@ -190,6 +192,44 @@ protected:
mTargets.push_back(streamRes);
}
+ {
+ MP3Resource res;
+ res.mFilePath = "small-shot.mp3";
+ res.mIsVBR = true;
+ res.mFileSize = 6825;
+ res.mMPEGLayer = 3;
+ res.mMPEGVersion = 1;
+ res.mID3MajorVersion = 4;
+ res.mID3MinorVersion = 0;
+ res.mID3Flags = 0;
+ res.mID3Size = 24;
+ res.mDuration = 336686;
+ res.mDurationError = 0.01f;
+ res.mSeekError = 0.2f;
+ res.mSampleRate = 44100;
+ res.mSamplesPerFrame = 1152;
+ res.mNumSamples = 12;
+ res.mNumTrailingFrames = 0;
+ res.mBitrate = 256000;
+ res.mSlotSize = 1;
+ res.mPrivate = 0;
+ const int syncs[] = { 34, 556, 1078, 1601, 2123, 2646, 3168, 3691, 4213,
+ 4736, 5258, 5781, 6303 };
+ res.mSyncOffsets.insert(res.mSyncOffsets.begin(), syncs, syncs + 13);
+
+ // No content length can be estimated for CBR stream resources.
+ MP3Resource streamRes = res;
+ streamRes.mFileSize = -1;
+
+ res.mResource = new MockMP3MediaResource(res.mFilePath);
+ res.mDemuxer = new MP3TrackDemuxer(res.mResource);
+ mTargets.push_back(res);
+
+ streamRes.mResource = new MockMP3StreamMediaResource(streamRes.mFilePath);
+ streamRes.mDemuxer = new MP3TrackDemuxer(streamRes.mResource);
+ mTargets.push_back(streamRes);
+ }
+
for (auto& target: mTargets) {
ASSERT_EQ(NS_OK, target.mResource->Open(nullptr));
ASSERT_TRUE(target.mDemuxer->Init());
@@ -307,11 +347,34 @@ TEST_F(MP3DemuxerTest, Duration) {
frameData = target.mDemuxer->DemuxSample();
}
}
+
+ // Seek out of range tests.
+ for (const auto& target: mTargets) {
+ // Skip tests for stream media resources because of lacking duration.
+ if (target.mFileSize <= 0) {
+ continue;
+ }
+
+ target.mDemuxer->Reset();
+ RefPtr frameData(target.mDemuxer->DemuxSample());
+ ASSERT_TRUE(frameData);
+
+ const int64_t duration = target.mDemuxer->Duration().ToMicroseconds();
+ const int64_t pos = duration + 1e6;
+
+ // Attempt to seek 1 second past the end of stream.
+ target.mDemuxer->Seek(TimeUnit::FromMicroseconds(pos));
+ // The seek should bring us to the end of the stream.
+ EXPECT_NEAR(duration, target.mDemuxer->SeekPosition().ToMicroseconds(),
+ target.mSeekError * duration);
+
+ // Since we're at the end of the stream, there should be no frames left.
+ frameData = target.mDemuxer->DemuxSample();
+ ASSERT_FALSE(frameData);
+ }
}
TEST_F(MP3DemuxerTest, Seek) {
- using media::TimeUnit;
-
for (const auto& target: mTargets) {
RefPtr frameData(target.mDemuxer->DemuxSample());
ASSERT_TRUE(frameData);
diff --git a/dom/media/gtest/moz.build b/dom/media/gtest/moz.build
index e287f1d7dd5..27c76dcb74b 100644
--- a/dom/media/gtest/moz.build
+++ b/dom/media/gtest/moz.build
@@ -48,6 +48,7 @@ TEST_HARNESS_FILES.gtest += [
'noise.mp3',
'noise_vbr.mp3',
'short-zero-in-moov.mp4',
+ 'small-shot.mp3',
'test.webm',
]
diff --git a/dom/media/gtest/small-shot.mp3 b/dom/media/gtest/small-shot.mp3
new file mode 100644
index 00000000000..f9397a5106b
Binary files /dev/null and b/dom/media/gtest/small-shot.mp3 differ
diff --git a/dom/media/test/crashtests/crashtests.list b/dom/media/test/crashtests/crashtests.list
index 63dbfe30c3b..df560ab2a0c 100644
--- a/dom/media/test/crashtests/crashtests.list
+++ b/dom/media/test/crashtests/crashtests.list
@@ -7,7 +7,7 @@ load 474744-1.html
HTTP load 481136-1.html # needs to be HTTP to recognize the ogg as an audio file?
load 492286-1.xhtml
load 493915-1.html
-load 495794-1.html
+skip-if(B2G) load 495794-1.html # in b2g all the media are muted by default
load 576612-1.html
load 752784-1.html
load 789075-1.html
diff --git a/dom/media/test/test_eme_playback.html b/dom/media/test/test_eme_playback.html
index 2823a6ae38a..798bad6d314 100644
--- a/dom/media/test/test_eme_playback.html
+++ b/dom/media/test/test_eme_playback.html
@@ -12,6 +12,19 @@
+
+
+
+
+
+
+
diff --git a/testing/web-platform/mozilla/tests/service-workers/service-worker/navigation-redirect.https.html b/testing/web-platform/mozilla/tests/service-workers/service-worker/navigation-redirect.https.html
new file mode 100644
index 00000000000..7b606cf0c3c
--- /dev/null
+++ b/testing/web-platform/mozilla/tests/service-workers/service-worker/navigation-redirect.https.html
@@ -0,0 +1,449 @@
+
+Service Worker: Navigation redirection
+
+
+
+
+
+
+
diff --git a/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/fetch-waits-for-activate-worker.js b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/fetch-waits-for-activate-worker.js
new file mode 100644
index 00000000000..66f3e593619
--- /dev/null
+++ b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/fetch-waits-for-activate-worker.js
@@ -0,0 +1,17 @@
+var activatePromiseResolve;
+
+addEventListener('activate', function(evt) {
+ evt.waitUntil(new Promise(function(resolve) {
+ activatePromiseResolve = resolve;
+ }));
+});
+
+addEventListener('message', function(evt) {
+ if (typeof activatePromiseResolve === 'function') {
+ activatePromiseResolve();
+ }
+});
+
+addEventListener('fetch', function(evt) {
+ evt.respondWith(new Response('Hello world'));
+});
diff --git a/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-other-origin.html b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-other-origin.html
new file mode 100644
index 00000000000..c1441ba685a
--- /dev/null
+++ b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-other-origin.html
@@ -0,0 +1,66 @@
+
+
+
+
diff --git a/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-out-scope.py b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-out-scope.py
new file mode 100644
index 00000000000..4b40762d89f
--- /dev/null
+++ b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-out-scope.py
@@ -0,0 +1,15 @@
+def main(request, response):
+ if "url" in request.GET:
+ headers = [("Location", request.GET["url"])]
+ return 302, headers, ''
+
+ return [], '''
+
+
+'''
diff --git a/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-scope1.py b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-scope1.py
new file mode 100644
index 00000000000..4b40762d89f
--- /dev/null
+++ b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-scope1.py
@@ -0,0 +1,15 @@
+def main(request, response):
+ if "url" in request.GET:
+ headers = [("Location", request.GET["url"])]
+ return 302, headers, ''
+
+ return [], '''
+
+
+'''
diff --git a/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-scope2.py b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-scope2.py
new file mode 100644
index 00000000000..4b40762d89f
--- /dev/null
+++ b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-scope2.py
@@ -0,0 +1,15 @@
+def main(request, response):
+ if "url" in request.GET:
+ headers = [("Location", request.GET["url"])]
+ return 302, headers, ''
+
+ return [], '''
+
+
+'''
diff --git a/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-worker.js b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-worker.js
new file mode 100644
index 00000000000..cb15b3ff113
--- /dev/null
+++ b/testing/web-platform/mozilla/tests/service-workers/service-worker/resources/navigation-redirect-worker.js
@@ -0,0 +1,75 @@
+// We store an empty response for each fetch event request we see
+// in this Cache object so we can get the list of urls in the
+// message event.
+var cacheName = 'urls-' + self.registration.scope;
+
+var waitUntilPromiseList = [];
+
+self.addEventListener('message', function(event) {
+ var urls;
+ event.waitUntil(Promise.all(waitUntilPromiseList).then(function() {
+ waitUntilPromiseList = [];
+ return caches.open(cacheName);
+ }).then(function(cache) {
+ return cache.keys();
+ }).then(function(requestList) {
+ urls = requestList.map(function(request) { return request.url; });
+ return caches.delete(cacheName);
+ }).then(function() {
+ event.data.port.postMessage({urls: urls});
+ }));
+ });
+
+function get_query_params(url) {
+ var search = (new URL(url)).search;
+ if (!search) {
+ return {};
+ }
+ var ret = {};
+ var params = search.substring(1).split('&');
+ params.forEach(function(param) {
+ var element = param.split('=');
+ ret[decodeURIComponent(element[0])] = decodeURIComponent(element[1]);
+ });
+ return ret;
+}
+
+self.addEventListener('fetch', function(event) {
+ var waitUntilPromise = caches.open(cacheName).then(function(cache) {
+ return cache.put(event.request, new Response());
+ });
+ event.waitUntil(waitUntilPromise);
+
+ var params = get_query_params(event.request.url);
+ if (!params['sw']) {
+ // To avoid races, add the waitUntil() promise to our global list.
+ // If we get a message event before we finish here, it will wait
+ // these promises to complete before proceeding to read from the
+ // cache.
+ waitUntilPromiseList.push(waitUntilPromise);
+ return;
+ }
+
+ event.respondWith(waitUntilPromise.then(function() {
+ if (params['sw'] == 'gen') {
+ return Response.redirect(params['url']);
+ } else if (params['sw'] == 'fetch') {
+ return fetch(event.request);
+ } else if (params['sw'] == 'opaque') {
+ return fetch(new Request(event.request.url, {redirect: 'manual'}));
+ } else if (params['sw'] == 'opaqueThroughCache') {
+ var url = event.request.url;
+ var cache;
+ return caches.delete(url)
+ .then(function() { return self.caches.open(url); })
+ .then(function(c) {
+ cache = c;
+ return fetch(new Request(url, {redirect: 'manual'}));
+ })
+ .then(function(res) { return cache.put(event.request, res); })
+ .then(function() { return cache.match(url); });
+ }
+
+ // unexpected... trigger an interception failure
+ }));
+ });
diff --git a/testing/web-platform/tests/web-animations/keyframe-effect/constructor.html b/testing/web-platform/tests/web-animations/keyframe-effect/constructor.html
index 2e549fef37f..3b786b50850 100644
--- a/testing/web-platform/tests/web-animations/keyframe-effect/constructor.html
+++ b/testing/web-platform/tests/web-animations/keyframe-effect/constructor.html
@@ -449,6 +449,181 @@ gKeyframeSequenceTests.forEach(function(subtest) {
" roundtrips");
});
+
+// KeyframeEffectOptions
+test(function(t) {
+ var effect = new KeyframeEffectReadOnly(
+ target, {left: ["10px", "20px"]});
+
+ var ct = effect.getComputedTiming();
+ assert_equals(ct.delay, 0, "default delay");
+ assert_equals(ct.fill, "none", "default fill");
+ assert_equals(ct.iterations, 1.0, "default iterations");
+ assert_equals(ct.duration, 0, "default duration");
+ assert_equals(ct.direction, "normal", "default direction");
+
+ assert_equals(effect.composite, "replace", "default composite");
+ assert_equals(effect.iterationComposite, "replace",
+ "default iterationComposite");
+ assert_equals(effect.spacing, "distribute",
+ "default spacing");
+}, "a KeyframeEffectReadOnly constructed without any " +
+ "KeyframeEffectOptions object");
+
+var gKeyframeEffectOptionTests = [
+ { desc: "an empty KeyframeEffectOption",
+ input: {},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "a normal KeyframeEffectOption",
+ input: {delay: 1000,
+ fill: "auto",
+ iterations: 5.5,
+ duration: "auto",
+ direction: "alternate"},
+ expected: {delay: 1000,
+ fill: "none",
+ iterations: 5.5,
+ duration: 0,
+ direction: "alternate"} },
+ { desc: "a double value",
+ input: 3000,
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 3000,
+ direction: "normal"} },
+ { desc: "+Infinity",
+ input: Infinity,
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: Infinity,
+ direction: "normal"} },
+ { desc: "-Infinity",
+ input: -Infinity,
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "NaN",
+ input: NaN,
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "a negative value",
+ input: -1,
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "an Infinity duration",
+ input: {duration: Infinity},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: Infinity,
+ direction: "normal"} },
+ { desc: "a negative Infinity duration",
+ input: {duration: -Infinity},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "a NaN duration",
+ input: {duration: NaN},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "a negative duration",
+ input: {duration: -1},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "a string duration",
+ input: {duration: "cabbage"},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "an auto duration",
+ input: {duration: "auto"},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "an Infinity iterations",
+ input: {iterations: Infinity},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: Infinity,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "a negative Infinity iterations",
+ input: {iterations: -Infinity},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "a NaN iterations",
+ input: {iterations: NaN},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "a negative iterations",
+ input: {iterations: -1},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "an auto fill",
+ input: {fill: "auto"},
+ expected: {delay: 0,
+ fill: "none",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} },
+ { desc: "a forwards fill",
+ input: {fill: "forwards"},
+ expected: {delay: 0,
+ fill: "forwards",
+ iterations: 1,
+ duration: 0,
+ direction: "normal"} }
+];
+
+gKeyframeEffectOptionTests.forEach(function(stest) {
+ test(function(t) {
+ var effect = new KeyframeEffectReadOnly(
+ target, {left: ["10px", "20px"]}, stest.input);
+
+ var ct = effect.getComputedTiming();
+ assert_equals(ct.delay, stest.expected.delay, "initial delay");
+ assert_equals(ct.fill, stest.expected.fill, "initial fill");
+ assert_equals(ct.iterations, stest.expected.iterations, "initial iterations");
+ assert_equals(ct.duration, stest.expected.duration, "initial duration");
+ assert_equals(ct.direction, stest.expected.direction, "initial direction");
+ }, "a KeyframeEffectReadOnly constructed by " + stest.desc);
+});
+
done();
diff --git a/toolkit/components/places/tests/cpp/mock_Link.h b/toolkit/components/places/tests/cpp/mock_Link.h
index 10c745fe5e6..7d2ec50a39d 100644
--- a/toolkit/components/places/tests/cpp/mock_Link.h
+++ b/toolkit/components/places/tests/cpp/mock_Link.h
@@ -117,18 +117,6 @@ Link::SizeOfExcludingThis(mozilla::MallocSizeOf aMallocSizeOf) const
return 0;
}
-void
-Link::URLSearchParamsUpdated(URLSearchParams* aSearchParams)
-{
- NS_NOTREACHED("Unexpected call to Link::URLSearchParamsUpdated");
-}
-
-void
-Link::UpdateURLSearchParams()
-{
- NS_NOTREACHED("Unexpected call to Link::UpdateURLSearchParams");
-}
-
NS_IMPL_CYCLE_COLLECTION_CLASS(URLSearchParams)
NS_IMPL_CYCLE_COLLECTION_UNLINK_BEGIN(URLSearchParams)
NS_IMPL_CYCLE_COLLECTION_UNLINK_END
diff --git a/toolkit/components/telemetry/Histograms.json b/toolkit/components/telemetry/Histograms.json
index 152500b9e52..2ded6691797 100644
--- a/toolkit/components/telemetry/Histograms.json
+++ b/toolkit/components/telemetry/Histograms.json
@@ -467,14 +467,14 @@
"kind": "flag",
"description": "XUL cache was disabled"
},
- "MEMORY_RESIDENT": {
+ "MEMORY_RESIDENT_FAST": {
"alert_emails": ["memshrink-telemetry-alerts@mozilla.com"],
"expires_in_version": "never",
"kind": "exponential",
"low": "32 * 1024",
"high": "16 * 1024 * 1024",
- "n_buckets": 200,
- "extended_statistics_ok": true,
+ "n_buckets": 100,
+ "bug_numbers": [1226196],
"description": "Resident memory size (KB)"
},
"MEMORY_UNIQUE": {
diff --git a/toolkit/components/telemetry/TelemetrySession.jsm b/toolkit/components/telemetry/TelemetrySession.jsm
index cf3c9024e86..418cee087f9 100644
--- a/toolkit/components/telemetry/TelemetrySession.jsm
+++ b/toolkit/components/telemetry/TelemetrySession.jsm
@@ -1041,7 +1041,7 @@ var Impl = {
b("MEMORY_VSIZE", "vsize");
b("MEMORY_VSIZE_MAX_CONTIGUOUS", "vsizeMaxContiguous");
- b("MEMORY_RESIDENT", "residentFast");
+ b("MEMORY_RESIDENT_FAST", "residentFast");
b("MEMORY_UNIQUE", "residentUnique");
b("MEMORY_HEAP_ALLOCATED", "heapAllocated");
p("MEMORY_HEAP_COMMITTED_UNUSED_RATIO", "heapOverheadRatio");
diff --git a/toolkit/components/telemetry/bucket-whitelist.json b/toolkit/components/telemetry/bucket-whitelist.json
index 62ec80501bf..5464acb28a8 100644
--- a/toolkit/components/telemetry/bucket-whitelist.json
+++ b/toolkit/components/telemetry/bucket-whitelist.json
@@ -1,5 +1,4 @@
[
- "MEMORY_RESIDENT",
"MEMORY_JS_MAIN_RUNTIME_TEMPORARY_PEAK",
"MEMORY_JS_GC_HEAP",
"MEMORY_HEAP_ALLOCATED",
diff --git a/toolkit/content/browser-child.js b/toolkit/content/browser-child.js
index 18c3930370e..327851e4896 100644
--- a/toolkit/content/browser-child.js
+++ b/toolkit/content/browser-child.js
@@ -361,7 +361,6 @@ addEventListener("DOMWindowClose", function (aEvent) {
if (!aEvent.isTrusted)
return;
sendAsyncMessage("DOMWindowClose");
- aEvent.preventDefault();
}, false);
addEventListener("ImageContentLoaded", function (aEvent) {
diff --git a/toolkit/content/widgets/videocontrols.xml b/toolkit/content/widgets/videocontrols.xml
index a7e49ebf73c..8b7f642199b 100644
--- a/toolkit/content/widgets/videocontrols.xml
+++ b/toolkit/content/widgets/videocontrols.xml
@@ -380,7 +380,7 @@
? this.video.readyState < this.video.HAVE_CURRENT_DATA
: this.video.readyState < this.video.HAVE_FUTURE_DATA)) ||
(this.timeUpdateCount <= 1 && !this.video.ended &&
- this.video.readyState < this.video.HAVE_ENOUGH_DATA &&
+ this.video.readyState < this.video.HAVE_FUTURE_DATA &&
this.video.networkState == this.video.NETWORK_LOADING))
show = true;
diff --git a/widget/android/nsWindow.cpp b/widget/android/nsWindow.cpp
index 267becc6f8d..5793c7cfa0c 100644
--- a/widget/android/nsWindow.cpp
+++ b/widget/android/nsWindow.cpp
@@ -2637,7 +2637,8 @@ nsWindow::DrawWindowUnderlay(LayerManagerComposite* aManager, nsIntRect aRect)
}
void
-nsWindow::DrawWindowOverlay(LayerManagerComposite* aManager, nsIntRect aRect)
+nsWindow::DrawWindowOverlay(LayerManagerComposite* aManager,
+ LayoutDeviceIntRect aRect)
{
PROFILER_LABEL("nsWindow", "DrawWindowOverlay",
js::ProfileEntry::Category::GRAPHICS);
diff --git a/widget/android/nsWindow.h b/widget/android/nsWindow.h
index 2cd4a2f8709..2c9c3c2a44c 100644
--- a/widget/android/nsWindow.h
+++ b/widget/android/nsWindow.h
@@ -152,7 +152,7 @@ public:
virtual bool NeedsPaint() override;
virtual void DrawWindowUnderlay(LayerManagerComposite* aManager, nsIntRect aRect) override;
- virtual void DrawWindowOverlay(LayerManagerComposite* aManager, nsIntRect aRect) override;
+ virtual void DrawWindowOverlay(LayerManagerComposite* aManager, LayoutDeviceIntRect aRect) override;
virtual mozilla::layers::CompositorParent* NewCompositorParent(int aSurfaceWidth, int aSurfaceHeight) override;
diff --git a/widget/cocoa/TextInputHandler.mm b/widget/cocoa/TextInputHandler.mm
index e08c8a0e3b0..7232b003614 100644
--- a/widget/cocoa/TextInputHandler.mm
+++ b/widget/cocoa/TextInputHandler.mm
@@ -3316,8 +3316,7 @@ IMEInputHandler::FirstRectForCharacterRange(NSRange& aRange,
if (!rootWindow || !rootView) {
return rect;
}
- rect = nsCocoaUtils::DevPixelsToCocoaPoints(r.ToUnknownRect(),
- mWidget->BackingScaleFactor());
+ rect = nsCocoaUtils::DevPixelsToCocoaPoints(r, mWidget->BackingScaleFactor());
rect = [rootView convertRect:rect toView:nil];
rect.origin = [rootWindow convertBaseToScreen:rect.origin];
diff --git a/widget/cocoa/VibrancyManager.mm b/widget/cocoa/VibrancyManager.mm
index bdadb1465f2..33f04a9e283 100644
--- a/widget/cocoa/VibrancyManager.mm
+++ b/widget/cocoa/VibrancyManager.mm
@@ -32,7 +32,7 @@ VibrancyManager::UpdateVibrantRegion(VibrancyType aType, const nsIntRegion& aReg
for (size_t i = 0; (iterRect = iter.Next()) || i < viewsToRecycle.Length(); ++i) {
if (iterRect) {
NSView* view = nil;
- NSRect rect = mCoordinateConverter.DevPixelsToCocoaPoints(*iterRect);
+ NSRect rect = mCoordinateConverter.UntypedDevPixelsToCocoaPoints(*iterRect);
if (i < viewsToRecycle.Length()) {
view = viewsToRecycle[i];
[view setFrame:rect];
@@ -72,7 +72,7 @@ VibrancyManager::ClearVibrantRegion(const VibrantRegion& aVibrantRegion) const
nsIntRegionRectIterator iter(aVibrantRegion.region);
while (const nsIntRect* rect = iter.Next()) {
- NSRectFill(mCoordinateConverter.DevPixelsToCocoaPoints(*rect));
+ NSRectFill(mCoordinateConverter.UntypedDevPixelsToCocoaPoints(*rect));
}
}
diff --git a/widget/cocoa/nsChildView.h b/widget/cocoa/nsChildView.h
index 747450a867e..37683f33857 100644
--- a/widget/cocoa/nsChildView.h
+++ b/widget/cocoa/nsChildView.h
@@ -403,7 +403,7 @@ public:
virtual void* GetNativeData(uint32_t aDataType) override;
virtual nsresult ConfigureChildren(const nsTArray& aConfigurations) override;
virtual LayoutDeviceIntPoint WidgetToScreenOffset() override;
- virtual bool ShowsResizeIndicator(nsIntRect* aResizerRect) override;
+ virtual bool ShowsResizeIndicator(LayoutDeviceIntRect* aResizerRect) override;
static bool ConvertStatus(nsEventStatus aStatus)
{ return aStatus == nsEventStatus_eConsumeNoDefault; }
@@ -487,12 +487,13 @@ public:
virtual void CleanupWindowEffects() override;
virtual bool PreRender(LayerManagerComposite* aManager) override;
virtual void PostRender(LayerManagerComposite* aManager) override;
- virtual void DrawWindowOverlay(LayerManagerComposite* aManager, nsIntRect aRect) override;
+ virtual void DrawWindowOverlay(LayerManagerComposite* aManager,
+ LayoutDeviceIntRect aRect) override;
virtual void UpdateThemeGeometries(const nsTArray& aThemeGeometries) override;
virtual void UpdateWindowDraggingRegion(const nsIntRegion& aRegion) override;
- const nsIntRegion& GetDraggableRegion() { return mDraggableRegion; }
+ const LayoutDeviceIntRegion& GetDraggableRegion() { return mDraggableRegion; }
virtual void ReportSwipeStarted(uint64_t aInputBlockId, bool aStartSwipe) override;
@@ -524,13 +525,18 @@ public:
nsIntPoint CocoaPointsToDevPixels(const NSPoint& aPt) const {
return nsCocoaUtils::CocoaPointsToDevPixels(aPt, BackingScaleFactor());
}
- nsIntRect CocoaPointsToDevPixels(const NSRect& aRect) const {
+ LayoutDeviceIntRect CocoaPointsToDevPixels(const NSRect& aRect) const {
return nsCocoaUtils::CocoaPointsToDevPixels(aRect, BackingScaleFactor());
}
CGFloat DevPixelsToCocoaPoints(int32_t aPixels) const {
return nsCocoaUtils::DevPixelsToCocoaPoints(aPixels, BackingScaleFactor());
}
- NSRect DevPixelsToCocoaPoints(const nsIntRect& aRect) const {
+ // XXX: all calls to this function should eventually be replaced with calls
+ // to DevPixelsToCocoaPoints().
+ NSRect UntypedDevPixelsToCocoaPoints(const nsIntRect& aRect) const {
+ return nsCocoaUtils::UntypedDevPixelsToCocoaPoints(aRect, BackingScaleFactor());
+ }
+ NSRect DevPixelsToCocoaPoints(const LayoutDeviceIntRect& aRect) const {
return nsCocoaUtils::DevPixelsToCocoaPoints(aRect, BackingScaleFactor());
}
@@ -577,19 +583,19 @@ protected:
void ConfigureAPZCTreeManager() override;
void ConfigureAPZControllerThread() override;
- void DoRemoteComposition(const nsIntRect& aRenderRect);
+ void DoRemoteComposition(const LayoutDeviceIntRect& aRenderRect);
// Overlay drawing functions for OpenGL drawing
- void DrawWindowOverlay(mozilla::layers::GLManager* aManager, nsIntRect aRect);
- void MaybeDrawResizeIndicator(mozilla::layers::GLManager* aManager, const nsIntRect& aRect);
- void MaybeDrawRoundedCorners(mozilla::layers::GLManager* aManager, const nsIntRect& aRect);
- void MaybeDrawTitlebar(mozilla::layers::GLManager* aManager, const nsIntRect& aRect);
+ void DrawWindowOverlay(mozilla::layers::GLManager* aManager, LayoutDeviceIntRect aRect);
+ void MaybeDrawResizeIndicator(mozilla::layers::GLManager* aManager);
+ void MaybeDrawRoundedCorners(mozilla::layers::GLManager* aManager, const LayoutDeviceIntRect& aRect);
+ void MaybeDrawTitlebar(mozilla::layers::GLManager* aManager);
// Redraw the contents of mTitlebarCGContext on the main thread, as
// determined by mDirtyTitlebarRegion.
void UpdateTitlebarCGContext();
- nsIntRect RectContainingTitlebarControls();
+ LayoutDeviceIntRect RectContainingTitlebarControls();
void UpdateVibrancy(const nsTArray& aThemeGeometries);
mozilla::VibrancyManager& EnsureVibrancyManager();
@@ -631,16 +637,16 @@ protected:
// May be accessed from any thread, protected
// by mEffectsLock.
bool mShowsResizeIndicator;
- nsIntRect mResizeIndicatorRect;
+ LayoutDeviceIntRect mResizeIndicatorRect;
bool mHasRoundedBottomCorners;
int mDevPixelCornerRadius;
bool mIsCoveringTitlebar;
bool mIsFullscreen;
- nsIntRect mTitlebarRect;
+ LayoutDeviceIntRect mTitlebarRect;
// The area of mTitlebarCGContext that needs to be redrawn during the next
// transaction. Accessed from any thread, protected by mEffectsLock.
- nsIntRegion mUpdatedTitlebarRegion;
+ LayoutDeviceIntRegion mUpdatedTitlebarRegion;
CGContextRef mTitlebarCGContext;
// Compositor thread only
@@ -653,7 +659,7 @@ protected:
// uploaded to to mTitlebarImage. Main thread only.
nsIntRegion mDirtyTitlebarRegion;
- nsIntRegion mDraggableRegion;
+ LayoutDeviceIntRegion mDraggableRegion;
// Cached value of [mView backingScaleFactor], to avoid sending two obj-c
// messages (respondsToSelector, backingScaleFactor) every time we need to
diff --git a/widget/cocoa/nsChildView.mm b/widget/cocoa/nsChildView.mm
index 126f289e99b..d7578af5cda 100644
--- a/widget/cocoa/nsChildView.mm
+++ b/widget/cocoa/nsChildView.mm
@@ -162,7 +162,7 @@ static uint32_t gNumberOfWidgetsNeedingEventThread = 0;
- (void)processPendingRedraws;
- (void)drawRect:(NSRect)aRect inContext:(CGContextRef)aContext;
-- (nsIntRegion)nativeDirtyRegionWithBoundingRect:(NSRect)aRect;
+- (LayoutDeviceIntRegion)nativeDirtyRegionWithBoundingRect:(NSRect)aRect;
- (BOOL)isUsingMainThreadOpenGL;
- (BOOL)isUsingOpenGL;
- (void)drawUsingOpenGL;
@@ -279,13 +279,15 @@ public:
virtual ~RectTextureImage();
already_AddRefed
- BeginUpdate(const nsIntSize& aNewSize,
- const nsIntRegion& aDirtyRegion = nsIntRegion());
+ BeginUpdate(const LayoutDeviceIntSize& aNewSize,
+ const LayoutDeviceIntRegion& aDirtyRegion =
+ LayoutDeviceIntRegion());
void EndUpdate(bool aKeepSurface = false);
- void UpdateIfNeeded(const nsIntSize& aNewSize,
- const nsIntRegion& aDirtyRegion,
- void (^aCallback)(gfx::DrawTarget*, const nsIntRegion&))
+ void UpdateIfNeeded(const LayoutDeviceIntSize& aNewSize,
+ const LayoutDeviceIntRegion& aDirtyRegion,
+ void (^aCallback)(gfx::DrawTarget*,
+ const LayoutDeviceIntRegion&))
{
RefPtr drawTarget = BeginUpdate(aNewSize, aDirtyRegion);
if (drawTarget) {
@@ -294,29 +296,30 @@ public:
}
}
- void UpdateFromCGContext(const nsIntSize& aNewSize,
- const nsIntRegion& aDirtyRegion,
+ void UpdateFromCGContext(const LayoutDeviceIntSize& aNewSize,
+ const LayoutDeviceIntRegion& aDirtyRegion,
CGContextRef aCGContext);
- nsIntRegion GetUpdateRegion() {
+ LayoutDeviceIntRegion GetUpdateRegion() {
MOZ_ASSERT(mInUpdate, "update region only valid during update");
return mUpdateRegion;
}
void Draw(mozilla::layers::GLManager* aManager,
- const nsIntPoint& aLocation,
+ const LayoutDeviceIntPoint& aLocation,
const Matrix4x4& aTransform = Matrix4x4());
- static nsIntSize TextureSizeForSize(const nsIntSize& aSize);
+ static LayoutDeviceIntSize TextureSizeForSize(
+ const LayoutDeviceIntSize& aSize);
protected:
RefPtr mUpdateDrawTarget;
GLContext* mGLContext;
- nsIntRegion mUpdateRegion;
- nsIntSize mUsedSize;
- nsIntSize mBufferSize;
- nsIntSize mTextureSize;
+ LayoutDeviceIntRegion mUpdateRegion;
+ LayoutDeviceIntSize mUsedSize;
+ LayoutDeviceIntSize mBufferSize;
+ LayoutDeviceIntSize mTextureSize;
GLuint mTexture;
bool mInUpdate;
};
@@ -356,7 +359,7 @@ public:
const gfx::Rect& aLayerRect,
const gfx::Rect& aTextureRect) override;
- void BeginFrame(nsIntSize aRenderSize);
+ void BeginFrame(LayoutDeviceIntSize aRenderSize);
void EndFrame();
NSOpenGLContext* GetNSOpenGLContext()
@@ -513,7 +516,8 @@ nsresult nsChildView::Create(nsIWidget *aParent,
// create our parallel NSView and hook it up to our parent. Recall
// that NS_NATIVE_WIDGET is the NSView.
CGFloat scaleFactor = nsCocoaUtils::GetBackingScaleFactor(mParentView);
- NSRect r = nsCocoaUtils::DevPixelsToCocoaPoints(mBounds, scaleFactor);
+ NSRect r = nsCocoaUtils::DevPixelsToCocoaPoints(
+ LayoutDeviceIntRect::FromUnknownRect(mBounds), scaleFactor);
mView = [(NSView*)CreateCocoaView(r) retain];
if (!mView) {
return NS_ERROR_FAILURE;
@@ -908,13 +912,9 @@ NS_IMETHODIMP nsChildView::SetCursor(imgIContainer* aCursor,
// Get this component dimension
NS_IMETHODIMP nsChildView::GetBounds(LayoutDeviceIntRect& aRect)
{
- nsIntRect tmp;
- if (!mView) {
- tmp = mBounds;
- } else {
- tmp = CocoaPointsToDevPixels([mView frame]);
- }
- aRect = LayoutDeviceIntRect::FromUnknownRect(tmp);
+ aRect = !mView
+ ? LayoutDeviceIntRect::FromUnknownRect(mBounds)
+ : CocoaPointsToDevPixels([mView frame]);
return NS_OK;
}
@@ -1006,7 +1006,7 @@ NS_IMETHODIMP nsChildView::Move(double aX, double aY)
mBounds.y = y;
ManipulateViewWithoutNeedingDisplay(mView, ^{
- [mView setFrame:DevPixelsToCocoaPoints(mBounds)];
+ [mView setFrame:UntypedDevPixelsToCocoaPoints(mBounds)];
});
NotifyRollupGeometryChange();
@@ -1031,7 +1031,7 @@ NS_IMETHODIMP nsChildView::Resize(double aWidth, double aHeight, bool aRepaint)
mBounds.height = height;
ManipulateViewWithoutNeedingDisplay(mView, ^{
- [mView setFrame:DevPixelsToCocoaPoints(mBounds)];
+ [mView setFrame:UntypedDevPixelsToCocoaPoints(mBounds)];
});
if (mVisible && aRepaint)
@@ -1070,7 +1070,7 @@ NS_IMETHODIMP nsChildView::Resize(double aX, double aY,
}
ManipulateViewWithoutNeedingDisplay(mView, ^{
- [mView setFrame:DevPixelsToCocoaPoints(mBounds)];
+ [mView setFrame:UntypedDevPixelsToCocoaPoints(mBounds)];
});
if (mVisible && aRepaint)
@@ -1092,7 +1092,7 @@ NS_IMETHODIMP nsChildView::Resize(double aX, double aY,
static const int32_t resizeIndicatorWidth = 15;
static const int32_t resizeIndicatorHeight = 15;
-bool nsChildView::ShowsResizeIndicator(nsIntRect* aResizerRect)
+bool nsChildView::ShowsResizeIndicator(LayoutDeviceIntRect* aResizerRect)
{
NSView *topLevelView = mView, *superView = nil;
while ((superView = [topLevelView superview]))
@@ -1364,10 +1364,10 @@ NS_IMETHODIMP nsChildView::Invalidate(const nsIntRect &aRect)
if ([NSView focusView]) {
// if a view is focussed (i.e. being drawn), then postpone the invalidate so that we
// don't lose it.
- [mView setNeedsPendingDisplayInRect:DevPixelsToCocoaPoints(aRect)];
+ [mView setNeedsPendingDisplayInRect:UntypedDevPixelsToCocoaPoints(aRect)];
}
else {
- [mView setNeedsDisplayInRect:DevPixelsToCocoaPoints(aRect)];
+ [mView setNeedsDisplayInRect:UntypedDevPixelsToCocoaPoints(aRect)];
}
return NS_OK;
@@ -1941,7 +1941,7 @@ nsChildView::ConfigureAPZControllerThread()
}
}
-nsIntRect
+LayoutDeviceIntRect
nsChildView::RectContainingTitlebarControls()
{
// Start with a thin strip at the top of the window for the highlight line.
@@ -2015,7 +2015,8 @@ nsChildView::PostRender(LayerManagerComposite* aManager)
}
void
-nsChildView::DrawWindowOverlay(LayerManagerComposite* aManager, nsIntRect aRect)
+nsChildView::DrawWindowOverlay(LayerManagerComposite* aManager,
+ LayoutDeviceIntRect aRect)
{
nsAutoPtr manager(GLManager::CreateGLManager(aManager));
if (manager) {
@@ -2024,20 +2025,20 @@ nsChildView::DrawWindowOverlay(LayerManagerComposite* aManager, nsIntRect aRect)
}
void
-nsChildView::DrawWindowOverlay(GLManager* aManager, nsIntRect aRect)
+nsChildView::DrawWindowOverlay(GLManager* aManager, LayoutDeviceIntRect aRect)
{
GLContext* gl = aManager->gl();
ScopedGLState scopedScissorTestState(gl, LOCAL_GL_SCISSOR_TEST, false);
- MaybeDrawTitlebar(aManager, aRect);
- MaybeDrawResizeIndicator(aManager, aRect);
+ MaybeDrawTitlebar(aManager);
+ MaybeDrawResizeIndicator(aManager);
MaybeDrawRoundedCorners(aManager, aRect);
}
static void
-ClearRegion(gfx::DrawTarget *aDT, nsIntRegion aRegion)
+ClearRegion(gfx::DrawTarget *aDT, LayoutDeviceIntRegion aRegion)
{
- gfxUtils::ClipToRegion(aDT, aRegion);
+ gfxUtils::ClipToRegion(aDT, aRegion.ToUnknownRegion());
aDT->ClearRect(gfx::Rect(0, 0, aDT->GetSize().width, aDT->GetSize().height));
aDT->PopClip();
}
@@ -2076,7 +2077,7 @@ DrawResizer(CGContextRef aCtx)
}
void
-nsChildView::MaybeDrawResizeIndicator(GLManager* aManager, const nsIntRect& aRect)
+nsChildView::MaybeDrawResizeIndicator(GLManager* aManager)
{
MutexAutoLock lock(mEffectsLock);
if (!mShowsResizeIndicator) {
@@ -2087,8 +2088,8 @@ nsChildView::MaybeDrawResizeIndicator(GLManager* aManager, const nsIntRect& aRec
mResizerImage = new RectTextureImage(aManager->gl());
}
- nsIntSize size = mResizeIndicatorRect.Size();
- mResizerImage->UpdateIfNeeded(size, nsIntRegion(), ^(gfx::DrawTarget* drawTarget, const nsIntRegion& updateRegion) {
+ LayoutDeviceIntSize size = mResizeIndicatorRect.Size();
+ mResizerImage->UpdateIfNeeded(size, LayoutDeviceIntRegion(), ^(gfx::DrawTarget* drawTarget, const LayoutDeviceIntRegion& updateRegion) {
ClearRegion(drawTarget, updateRegion);
gfx::BorrowedCGContext borrow(drawTarget);
DrawResizer(borrow.cg);
@@ -2131,7 +2132,7 @@ DrawTitlebarHighlight(NSSize aWindowSize, CGFloat aRadius, CGFloat aDevicePixelW
}
static CGContextRef
-CreateCGContext(const nsIntSize& aSize)
+CreateCGContext(const LayoutDeviceIntSize& aSize)
{
CGColorSpaceRef cs = CGColorSpaceCreateDeviceRGB();
CGContextRef ctx =
@@ -2164,7 +2165,8 @@ nsChildView::UpdateTitlebarCGContext()
NSRect dirtyRect = [mView convertRect:[(BaseWindow*)[mView window] getAndResetNativeDirtyRect] fromView:nil];
NSRect dirtyTitlebarRect = NSIntersectionRect(titlebarRect, dirtyRect);
- nsIntSize texSize = RectTextureImage::TextureSizeForSize(mTitlebarRect.Size());
+ LayoutDeviceIntSize texSize =
+ RectTextureImage::TextureSizeForSize(mTitlebarRect.Size());
if (!mTitlebarCGContext ||
CGBitmapContextGetWidth(mTitlebarCGContext) != size_t(texSize.width) ||
CGBitmapContextGetHeight(mTitlebarCGContext) != size_t(texSize.height)) {
@@ -2285,14 +2287,14 @@ nsChildView::UpdateTitlebarCGContext()
// GLContext surface. In order to make the titlebar controls visible, we have
// to redraw them inside the OpenGL context surface.
void
-nsChildView::MaybeDrawTitlebar(GLManager* aManager, const nsIntRect& aRect)
+nsChildView::MaybeDrawTitlebar(GLManager* aManager)
{
MutexAutoLock lock(mEffectsLock);
if (!mIsCoveringTitlebar || mIsFullscreen) {
return;
}
- nsIntRegion updatedTitlebarRegion;
+ LayoutDeviceIntRegion updatedTitlebarRegion;
updatedTitlebarRegion.And(mUpdatedTitlebarRegion, mTitlebarRect);
mUpdatedTitlebarRegion.SetEmpty();
@@ -2315,7 +2317,8 @@ DrawTopLeftCornerMask(CGContextRef aCtx, int aRadius)
}
void
-nsChildView::MaybeDrawRoundedCorners(GLManager* aManager, const nsIntRect& aRect)
+nsChildView::MaybeDrawRoundedCorners(GLManager* aManager,
+ const LayoutDeviceIntRect& aRect)
{
MutexAutoLock lock(mEffectsLock);
@@ -2323,8 +2326,8 @@ nsChildView::MaybeDrawRoundedCorners(GLManager* aManager, const nsIntRect& aRect
mCornerMaskImage = new RectTextureImage(aManager->gl());
}
- nsIntSize size(mDevPixelCornerRadius, mDevPixelCornerRadius);
- mCornerMaskImage->UpdateIfNeeded(size, nsIntRegion(), ^(gfx::DrawTarget* drawTarget, const nsIntRegion& updateRegion) {
+ LayoutDeviceIntSize size(mDevPixelCornerRadius, mDevPixelCornerRadius);
+ mCornerMaskImage->UpdateIfNeeded(size, LayoutDeviceIntRegion(), ^(gfx::DrawTarget* drawTarget, const LayoutDeviceIntRegion& updateRegion) {
ClearRegion(drawTarget, updateRegion);
RefPtr builder = drawTarget->CreatePathBuilder();
builder->Arc(gfx::Point(mDevPixelCornerRadius, mDevPixelCornerRadius), mDevPixelCornerRadius, 0, 2.0f * M_PI);
@@ -2392,17 +2395,17 @@ FindUnifiedToolbarBottom(const nsTArray& aThemeGeometr
return unifiedToolbarBottom;
}
-static nsIntRect
+static LayoutDeviceIntRect
FindFirstRectOfType(const nsTArray& aThemeGeometries,
nsITheme::ThemeGeometryType aThemeGeometryType)
{
for (uint32_t i = 0; i < aThemeGeometries.Length(); ++i) {
const nsIWidget::ThemeGeometry& g = aThemeGeometries[i];
if (g.mType == aThemeGeometryType) {
- return g.mRect;
+ return LayoutDeviceIntRect::FromUnknownRect(g.mRect);
}
}
- return nsIntRect();
+ return LayoutDeviceIntRect();
}
void
@@ -2434,9 +2437,9 @@ nsChildView::UpdateThemeGeometries(const nsTArray& aThemeGeometri
[win setSheetAttachmentPosition:DevPixelsToCocoaPoints(devSheetPosition)];
// Update titlebar control offsets.
- nsIntRect windowButtonRect = FindFirstRectOfType(aThemeGeometries, nsNativeThemeCocoa::eThemeGeometryTypeWindowButtons);
+ LayoutDeviceIntRect windowButtonRect = FindFirstRectOfType(aThemeGeometries, nsNativeThemeCocoa::eThemeGeometryTypeWindowButtons);
[win placeWindowButtons:[mView convertRect:DevPixelsToCocoaPoints(windowButtonRect) toView:nil]];
- nsIntRect fullScreenButtonRect = FindFirstRectOfType(aThemeGeometries, nsNativeThemeCocoa::eThemeGeometryTypeFullscreenButton);
+ LayoutDeviceIntRect fullScreenButtonRect = FindFirstRectOfType(aThemeGeometries, nsNativeThemeCocoa::eThemeGeometryTypeFullscreenButton);
[win placeFullScreenButton:[mView convertRect:DevPixelsToCocoaPoints(fullScreenButtonRect) toView:nil]];
}
@@ -2632,8 +2635,9 @@ nsChildView::StartRemoteDrawing()
}
}
- nsIntRegion dirtyRegion = mBounds;
- nsIntSize renderSize = mBounds.Size();
+ LayoutDeviceIntRegion dirtyRegion(LayoutDeviceIntRect::FromUnknownRect(mBounds));
+ LayoutDeviceIntSize renderSize =
+ LayoutDeviceIntSize::FromUnknownSize(mBounds.Size());
if (!mBasicCompositorImage) {
mBasicCompositorImage = new RectTextureImage(mGLPresenter->gl());
@@ -2644,7 +2648,7 @@ nsChildView::StartRemoteDrawing()
if (!drawTarget) {
// Composite unchanged textures.
- DoRemoteComposition(mBounds);
+ DoRemoteComposition(LayoutDeviceIntRect::FromUnknownRect(mBounds));
return nullptr;
}
@@ -2655,7 +2659,7 @@ void
nsChildView::EndRemoteDrawing()
{
mBasicCompositorImage->EndUpdate(true);
- DoRemoteComposition(mBounds);
+ DoRemoteComposition(LayoutDeviceIntRect::FromUnknownRect(mBounds));
}
void
@@ -2682,7 +2686,7 @@ nsChildView::InitCompositor(Compositor* aCompositor)
}
void
-nsChildView::DoRemoteComposition(const nsIntRect& aRenderRect)
+nsChildView::DoRemoteComposition(const LayoutDeviceIntRect& aRenderRect)
{
if (![(ChildView*)mView preRender:mGLPresenter->GetNSOpenGLContext()]) {
return;
@@ -2690,7 +2694,7 @@ nsChildView::DoRemoteComposition(const nsIntRect& aRenderRect)
mGLPresenter->BeginFrame(aRenderRect.Size());
// Draw the result from the basic compositor.
- mBasicCompositorImage->Draw(mGLPresenter, nsIntPoint(0, 0));
+ mBasicCompositorImage->Draw(mGLPresenter, LayoutDeviceIntPoint(0, 0));
// DrawWindowOverlay doesn't do anything for non-GL, so it didn't paint
// anything during the basic compositor transaction. Draw the overlay now.
@@ -2704,8 +2708,10 @@ nsChildView::DoRemoteComposition(const nsIntRect& aRenderRect)
void
nsChildView::UpdateWindowDraggingRegion(const nsIntRegion& aRegion)
{
- if (mDraggableRegion != aRegion) {
- mDraggableRegion = aRegion;
+ LayoutDeviceIntRegion region =
+ LayoutDeviceIntRegion::FromUnknownRegion(aRegion);
+ if (mDraggableRegion != region) {
+ mDraggableRegion = region;
[(ChildView*)mView updateWindowDraggableState];
}
}
@@ -2875,29 +2881,30 @@ RectTextureImage::~RectTextureImage()
}
}
-nsIntSize
-RectTextureImage::TextureSizeForSize(const nsIntSize& aSize)
+LayoutDeviceIntSize
+RectTextureImage::TextureSizeForSize(const LayoutDeviceIntSize& aSize)
{
- return nsIntSize(gfx::NextPowerOfTwo(aSize.width),
- gfx::NextPowerOfTwo(aSize.height));
+ return LayoutDeviceIntSize(gfx::NextPowerOfTwo(aSize.width),
+ gfx::NextPowerOfTwo(aSize.height));
}
already_AddRefed
-RectTextureImage::BeginUpdate(const nsIntSize& aNewSize,
- const nsIntRegion& aDirtyRegion)
+RectTextureImage::BeginUpdate(const LayoutDeviceIntSize& aNewSize,
+ const LayoutDeviceIntRegion& aDirtyRegion)
{
MOZ_ASSERT(!mInUpdate, "Beginning update during update!");
mUpdateRegion = aDirtyRegion;
if (aNewSize != mUsedSize) {
mUsedSize = aNewSize;
- mUpdateRegion = gfx::IntRect(gfx::IntPoint(0, 0), aNewSize);
+ mUpdateRegion =
+ LayoutDeviceIntRect(LayoutDeviceIntPoint(0, 0), aNewSize);
}
if (mUpdateRegion.IsEmpty()) {
return nullptr;
}
- nsIntSize neededBufferSize = TextureSizeForSize(mUsedSize);
+ LayoutDeviceIntSize neededBufferSize = TextureSizeForSize(mUsedSize);
if (!mUpdateDrawTarget || mBufferSize != neededBufferSize) {
gfx::IntSize size(neededBufferSize.width, neededBufferSize.height);
mUpdateDrawTarget =
@@ -2926,14 +2933,15 @@ RectTextureImage::EndUpdate(bool aKeepSurface)
MOZ_ASSERT(mInUpdate, "Ending update while not in update");
bool overwriteTexture = false;
- nsIntRegion updateRegion = mUpdateRegion;
+ LayoutDeviceIntRegion updateRegion = mUpdateRegion;
if (!mTexture || (mTextureSize != mBufferSize)) {
overwriteTexture = true;
mTextureSize = mBufferSize;
}
if (overwriteTexture || !CanUploadSubtextures()) {
- updateRegion = gfx::IntRect(gfx::IntPoint(0, 0), mTextureSize);
+ updateRegion =
+ LayoutDeviceIntRect(LayoutDeviceIntPoint(0, 0), mTextureSize);
}
RefPtr snapshot = mUpdateDrawTarget->Snapshot();
@@ -2941,10 +2949,10 @@ RectTextureImage::EndUpdate(bool aKeepSurface)
UploadSurfaceToTexture(mGLContext,
dataSnapshot,
- updateRegion,
+ updateRegion.ToUnknownRegion(),
mTexture,
overwriteTexture,
- updateRegion.GetBounds().TopLeft(),
+ updateRegion.GetBounds().TopLeft().ToUnknownPoint(),
false,
LOCAL_GL_TEXTURE0,
LOCAL_GL_TEXTURE_RECTANGLE_ARB);
@@ -2957,8 +2965,8 @@ RectTextureImage::EndUpdate(bool aKeepSurface)
}
void
-RectTextureImage::UpdateFromCGContext(const nsIntSize& aNewSize,
- const nsIntRegion& aDirtyRegion,
+RectTextureImage::UpdateFromCGContext(const LayoutDeviceIntSize& aNewSize,
+ const LayoutDeviceIntRegion& aDirtyRegion,
CGContextRef aCGContext)
{
gfx::IntSize size = gfx::IntSize(CGBitmapContextGetWidth(aCGContext),
@@ -2967,7 +2975,7 @@ RectTextureImage::UpdateFromCGContext(const nsIntSize& aNewSize,
RefPtr dt = BeginUpdate(aNewSize, aDirtyRegion);
if (dt) {
gfx::Rect rect(0, 0, size.width, size.height);
- gfxUtils::ClipToRegion(dt, GetUpdateRegion());
+ gfxUtils::ClipToRegion(dt, GetUpdateRegion().ToUnknownRegion());
RefPtr sourceSurface =
dt->CreateSourceSurfaceFromData(static_cast(CGBitmapContextGetData(aCGContext)),
size,
@@ -2983,7 +2991,7 @@ RectTextureImage::UpdateFromCGContext(const nsIntSize& aNewSize,
void
RectTextureImage::Draw(GLManager* aManager,
- const nsIntPoint& aLocation,
+ const LayoutDeviceIntPoint& aLocation,
const Matrix4x4& aTransform)
{
ShaderProgramOGL* program = aManager->GetProgram(LOCAL_GL_TEXTURE_RECTANGLE_ARB,
@@ -3076,7 +3084,7 @@ GLPresenter::BindAndDrawQuad(ShaderProgramOGL *aProgram,
}
void
-GLPresenter::BeginFrame(nsIntSize aRenderSize)
+GLPresenter::BeginFrame(LayoutDeviceIntSize aRenderSize)
{
mGLContext->MakeCurrent();
@@ -3610,9 +3618,9 @@ NSEvent* gLastDragMouseDownEvent = nil;
return mGeckoChild->VibrancyFontSmoothingBackgroundColorForThemeGeometryType(aThemeGeometryType);
}
-- (nsIntRegion)nativeDirtyRegionWithBoundingRect:(NSRect)aRect
+- (LayoutDeviceIntRegion)nativeDirtyRegionWithBoundingRect:(NSRect)aRect
{
- nsIntRect boundingRect = mGeckoChild->CocoaPointsToDevPixels(aRect);
+ LayoutDeviceIntRect boundingRect = mGeckoChild->CocoaPointsToDevPixels(aRect);
const NSRect *rects;
NSInteger count;
[self getRectsBeingDrawn:&rects count:&count];
@@ -3621,7 +3629,7 @@ NSEvent* gLastDragMouseDownEvent = nil;
return boundingRect;
}
- nsIntRegion region;
+ LayoutDeviceIntRegion region;
for (NSInteger i = 0; i < count; ++i) {
region.Or(region, mGeckoChild->CocoaPointsToDevPixels(rects[i]));
}
@@ -3698,7 +3706,7 @@ NSEvent* gLastDragMouseDownEvent = nil;
CGContextSaveGState(aContext);
- nsIntRegion region = [self nativeDirtyRegionWithBoundingRect:aRect];
+ LayoutDeviceIntRegion region = [self nativeDirtyRegionWithBoundingRect:aRect];
// Create Cairo objects.
RefPtr targetSurface;
@@ -3712,10 +3720,10 @@ NSEvent* gLastDragMouseDownEvent = nil;
RefPtr targetContext = new gfxContext(dt);
// Set up the clip region.
- nsIntRegionRectIterator iter(region);
+ LayoutDeviceIntRegion::RectIterator iter(region);
targetContext->NewPath();
for (;;) {
- const nsIntRect* r = iter.Next();
+ const LayoutDeviceIntRect* r = iter.Next();
if (!r)
break;
targetContext->Rectangle(gfxRect(r->x, r->y, r->width, r->height));
@@ -3727,10 +3735,10 @@ NSEvent* gLastDragMouseDownEvent = nil;
if (mGeckoChild->GetLayerManager()->GetBackendType() == LayersBackend::LAYERS_BASIC) {
nsBaseWidget::AutoLayerManagerSetup
setupLayerManager(mGeckoChild, targetContext, BufferMode::BUFFER_NONE);
- painted = mGeckoChild->PaintWindow(region);
+ painted = mGeckoChild->PaintWindow(region.ToUnknownRegion());
} else if (mGeckoChild->GetLayerManager()->GetBackendType() == LayersBackend::LAYERS_CLIENT) {
// We only need this so that we actually get DidPaintWindow fired
- painted = mGeckoChild->PaintWindow(region);
+ painted = mGeckoChild->PaintWindow(region.ToUnknownRegion());
}
targetContext = nullptr;
@@ -4588,15 +4596,16 @@ NSEvent* gLastDragMouseDownEvent = nil;
}
static CGSRegionObj
-NewCGSRegionFromRegion(const nsIntRegion& aRegion,
- CGRect (^aRectConverter)(const nsIntRect&))
+NewCGSRegionFromRegion(const LayoutDeviceIntRegion& aRegion,
+ CGRect (^aRectConverter)(const LayoutDeviceIntRect&))
{
nsTArray rects;
- nsIntRegionRectIterator iter(aRegion);
+ LayoutDeviceIntRegion::RectIterator iter(aRegion);
for (;;) {
- const nsIntRect* r = iter.Next();
- if (!r)
+ const LayoutDeviceIntRect* r = iter.Next();
+ if (!r) {
break;
+ }
rects.AppendElement(aRectConverter(*r));
}
@@ -4615,12 +4624,12 @@ NewCGSRegionFromRegion(const nsIntRegion& aRegion,
return [super _regionForOpaqueDescendants:aRect forMove:aForMove];
}
- nsIntRect boundingRect = mGeckoChild->CocoaPointsToDevPixels(aRect);
+ LayoutDeviceIntRect boundingRect = mGeckoChild->CocoaPointsToDevPixels(aRect);
- nsIntRegion opaqueRegion;
+ LayoutDeviceIntRegion opaqueRegion;
opaqueRegion.Sub(boundingRect, mGeckoChild->GetDraggableRegion());
- return NewCGSRegionFromRegion(opaqueRegion, ^(const nsIntRect& r) {
+ return NewCGSRegionFromRegion(opaqueRegion, ^(const LayoutDeviceIntRect& r) {
return [self convertToFlippedWindowCoordinates:mGeckoChild->DevPixelsToCocoaPoints(r)];
});
}
diff --git a/widget/cocoa/nsClipboard.mm b/widget/cocoa/nsClipboard.mm
index abc7031e88f..870638ac7e1 100644
--- a/widget/cocoa/nsClipboard.mm
+++ b/widget/cocoa/nsClipboard.mm
@@ -153,7 +153,12 @@ nsClipboard::TransferableFromPasteboard(nsITransferable *aTransferable, NSPasteb
if (!pString)
continue;
- NSData* stringData = [pString dataUsingEncoding:NSUnicodeStringEncoding];
+ NSData* stringData;
+ if ([pboardType isEqualToString:NSRTFPboardType]) {
+ stringData = [pString dataUsingEncoding:NSASCIIStringEncoding];
+ } else {
+ stringData = [pString dataUsingEncoding:NSUnicodeStringEncoding];
+ }
unsigned int dataLength = [stringData length];
void* clipboardDataPtr = malloc(dataLength);
if (!clipboardDataPtr)
@@ -581,12 +586,14 @@ nsClipboard::PasteboardDictFromTransferable(nsITransferable* aTransferable)
bool nsClipboard::IsStringType(const nsCString& aMIMEType, NSString** aPasteboardType)
{
- if (aMIMEType.EqualsLiteral(kUnicodeMime) ||
- aMIMEType.EqualsLiteral(kHTMLMime)) {
- if (aMIMEType.EqualsLiteral(kUnicodeMime))
- *aPasteboardType = NSStringPboardType;
- else
- *aPasteboardType = NSHTMLPboardType;
+ if (aMIMEType.EqualsLiteral(kUnicodeMime)) {
+ *aPasteboardType = NSStringPboardType;
+ return true;
+ } else if (aMIMEType.EqualsLiteral(kRTFMime)) {
+ *aPasteboardType = NSRTFPboardType;
+ return true;
+ } else if (aMIMEType.EqualsLiteral(kHTMLMime)) {
+ *aPasteboardType = NSHTMLPboardType;
return true;
} else {
return false;
diff --git a/widget/cocoa/nsCocoaUtils.h b/widget/cocoa/nsCocoaUtils.h
index e5a13120ab9..287cec0b448 100644
--- a/widget/cocoa/nsCocoaUtils.h
+++ b/widget/cocoa/nsCocoaUtils.h
@@ -129,6 +129,7 @@ struct KeyBindingsCommand
class nsCocoaUtils
{
typedef mozilla::gfx::SourceSurface SourceSurface;
+ typedef mozilla::LayoutDeviceIntRect LayoutDeviceIntRect;
public:
@@ -159,13 +160,13 @@ public:
NSToIntRound(aPt.y * aBackingScale));
}
- static nsIntRect
+ static LayoutDeviceIntRect
CocoaPointsToDevPixels(const NSRect& aRect, CGFloat aBackingScale)
{
- return nsIntRect(NSToIntRound(aRect.origin.x * aBackingScale),
- NSToIntRound(aRect.origin.y * aBackingScale),
- NSToIntRound(aRect.size.width * aBackingScale),
- NSToIntRound(aRect.size.height * aBackingScale));
+ return LayoutDeviceIntRect(NSToIntRound(aRect.origin.x * aBackingScale),
+ NSToIntRound(aRect.origin.y * aBackingScale),
+ NSToIntRound(aRect.size.width * aBackingScale),
+ NSToIntRound(aRect.size.height * aBackingScale));
}
static CGFloat
@@ -182,8 +183,20 @@ public:
(CGFloat)aPt.y / aBackingScale);
}
+ // XXX: all calls to this function should eventually be replaced with calls
+ // to DevPixelsToCocoaPoints().
static NSRect
- DevPixelsToCocoaPoints(const nsIntRect& aRect, CGFloat aBackingScale)
+ UntypedDevPixelsToCocoaPoints(const nsIntRect& aRect, CGFloat aBackingScale)
+ {
+ return NSMakeRect((CGFloat)aRect.x / aBackingScale,
+ (CGFloat)aRect.y / aBackingScale,
+ (CGFloat)aRect.width / aBackingScale,
+ (CGFloat)aRect.height / aBackingScale);
+ }
+
+ static NSRect
+ DevPixelsToCocoaPoints(const LayoutDeviceIntRect& aRect,
+ CGFloat aBackingScale)
{
return NSMakeRect((CGFloat)aRect.x / aBackingScale,
(CGFloat)aRect.y / aBackingScale,
@@ -217,8 +230,8 @@ public:
// See explanation for geckoRectToCocoaRect, guess what this does...
static nsIntRect CocoaRectToGeckoRect(const NSRect &cocoaRect);
- static nsIntRect CocoaRectToGeckoRectDevPix(const NSRect &aCocoaRect,
- CGFloat aBackingScale);
+ static mozilla::LayoutDeviceIntRect CocoaRectToGeckoRectDevPix(
+ const NSRect& aCocoaRect, CGFloat aBackingScale);
// Gives the location for the event in screen coordinates. Do not call this
// unless the window the event was originally targeted at is still alive!
diff --git a/widget/cocoa/nsCocoaUtils.mm b/widget/cocoa/nsCocoaUtils.mm
index 8a3adb2cc84..fa475a1d856 100644
--- a/widget/cocoa/nsCocoaUtils.mm
+++ b/widget/cocoa/nsCocoaUtils.mm
@@ -97,10 +97,10 @@ nsIntRect nsCocoaUtils::CocoaRectToGeckoRect(const NSRect &cocoaRect)
return rect;
}
-nsIntRect nsCocoaUtils::CocoaRectToGeckoRectDevPix(const NSRect &aCocoaRect,
- CGFloat aBackingScale)
+LayoutDeviceIntRect nsCocoaUtils::CocoaRectToGeckoRectDevPix(
+ const NSRect& aCocoaRect, CGFloat aBackingScale)
{
- nsIntRect rect;
+ LayoutDeviceIntRect rect;
rect.x = NSToIntRound(aCocoaRect.origin.x * aBackingScale);
rect.y = NSToIntRound(FlippedScreenY(aCocoaRect.origin.y + aCocoaRect.size.height) * aBackingScale);
rect.width = NSToIntRound((aCocoaRect.origin.x + aCocoaRect.size.width) * aBackingScale) - rect.x;
diff --git a/widget/cocoa/nsCocoaWindow.mm b/widget/cocoa/nsCocoaWindow.mm
index 5adf27fd87d..7964805a219 100644
--- a/widget/cocoa/nsCocoaWindow.mm
+++ b/widget/cocoa/nsCocoaWindow.mm
@@ -1573,8 +1573,7 @@ NS_IMETHODIMP nsCocoaWindow::GetClientBounds(mozilla::LayoutDeviceIntRect& aRect
CGFloat scaleFactor = BackingScaleFactor();
if (!mWindow) {
- aRect = LayoutDeviceIntRect::FromUnknownRect(
- nsCocoaUtils::CocoaRectToGeckoRectDevPix(NSZeroRect, scaleFactor));
+ aRect = nsCocoaUtils::CocoaRectToGeckoRectDevPix(NSZeroRect, scaleFactor);
return NS_OK;
}
@@ -1586,8 +1585,7 @@ NS_IMETHODIMP nsCocoaWindow::GetClientBounds(mozilla::LayoutDeviceIntRect& aRect
r = [mWindow contentRectForFrameRect:[mWindow frame]];
}
- aRect = LayoutDeviceIntRect::FromUnknownRect(
- nsCocoaUtils::CocoaRectToGeckoRectDevPix(r, scaleFactor));
+ aRect = nsCocoaUtils::CocoaRectToGeckoRectDevPix(r, scaleFactor);
return NS_OK;
@@ -1601,7 +1599,8 @@ nsCocoaWindow::UpdateBounds()
if (mWindow) {
frame = [mWindow frame];
}
- mBounds = nsCocoaUtils::CocoaRectToGeckoRectDevPix(frame, BackingScaleFactor());
+ mBounds = nsCocoaUtils::CocoaRectToGeckoRectDevPix(
+ frame, BackingScaleFactor()).ToUnknownRect();
}
NS_IMETHODIMP nsCocoaWindow::GetScreenBounds(LayoutDeviceIntRect &aRect)
@@ -1609,8 +1608,8 @@ NS_IMETHODIMP nsCocoaWindow::GetScreenBounds(LayoutDeviceIntRect &aRect)
NS_OBJC_BEGIN_TRY_ABORT_BLOCK_NSRESULT;
#ifdef DEBUG
- nsIntRect r = nsCocoaUtils::CocoaRectToGeckoRectDevPix([mWindow frame], BackingScaleFactor());
- NS_ASSERTION(mWindow && mBounds == r, "mBounds out of sync!");
+ LayoutDeviceIntRect r = nsCocoaUtils::CocoaRectToGeckoRectDevPix([mWindow frame], BackingScaleFactor());
+ NS_ASSERTION(mWindow && mBounds == r.ToUnknownRect(), "mBounds out of sync!");
#endif
aRect = LayoutDeviceIntRect::FromUnknownRect(mBounds);
@@ -1986,13 +1985,13 @@ LayoutDeviceIntPoint nsCocoaWindow::WidgetToScreenOffset()
NS_OBJC_BEGIN_TRY_ABORT_BLOCK_RETURN;
NSRect rect = NSZeroRect;
- nsIntRect r;
+ LayoutDeviceIntRect r;
if (mWindow) {
rect = [mWindow contentRectForFrameRect:[mWindow frame]];
}
r = nsCocoaUtils::CocoaRectToGeckoRectDevPix(rect, BackingScaleFactor());
- return LayoutDeviceIntPoint::FromUnknownPoint(r.TopLeft());
+ return r.TopLeft();
NS_OBJC_END_TRY_ABORT_BLOCK_RETURN(LayoutDeviceIntPoint(0,0));
}
@@ -2019,12 +2018,12 @@ nsCocoaWindow::ClientToWindowSize(const LayoutDeviceIntSize& aClientSize)
return LayoutDeviceIntSize(0, 0);
CGFloat backingScale = BackingScaleFactor();
- nsIntRect r(0, 0, aClientSize.width, aClientSize.height);
+ LayoutDeviceIntRect r(0, 0, aClientSize.width, aClientSize.height);
NSRect rect = nsCocoaUtils::DevPixelsToCocoaPoints(r, backingScale);
NSRect inflatedRect = [mWindow frameRectForContentRect:rect];
r = nsCocoaUtils::CocoaRectToGeckoRectDevPix(inflatedRect, backingScale);
- return LayoutDeviceIntSize(r.width, r.height);
+ return r.Size();
NS_OBJC_END_TRY_ABORT_BLOCK_RETURN(LayoutDeviceIntSize(0,0));
}
diff --git a/widget/cocoa/nsDragService.mm b/widget/cocoa/nsDragService.mm
index 6618ff4abea..121fb131c8f 100644
--- a/widget/cocoa/nsDragService.mm
+++ b/widget/cocoa/nsDragService.mm
@@ -493,9 +493,16 @@ nsDragService::GetData(nsITransferable* aTransferable, uint32_t aItemIndex)
pString = GetStringForType(item, (const NSString*)kUTTypeURL);
} else if (flavorStr.EqualsLiteral(kURLDescriptionMime)) {
pString = GetTitleForURL(item);
+ } else if (flavorStr.EqualsLiteral(kRTFMime)) {
+ pString = GetStringForType(item, (const NSString*)kUTTypeRTF);
}
if (pString) {
- NSData* stringData = [pString dataUsingEncoding:NSUnicodeStringEncoding];
+ NSData* stringData;
+ if (flavorStr.EqualsLiteral(kRTFMime)) {
+ stringData = [pString dataUsingEncoding:NSASCIIStringEncoding];
+ } else {
+ stringData = [pString dataUsingEncoding:NSUnicodeStringEncoding];
+ }
unsigned int dataLength = [stringData length];
void* clipboardDataPtr = malloc(dataLength);
if (!clipboardDataPtr)
@@ -601,6 +608,8 @@ nsDragService::IsDataFlavorSupported(const char *aDataFlavor, bool *_retval)
type = (const NSString*)kUTTypeURL;
} else if (dataFlavor.EqualsLiteral(kURLDescriptionMime)) {
type = (const NSString*)kUTTypeURLName;
+ } else if (dataFlavor.EqualsLiteral(kRTFMime)) {
+ type = (const NSString*)kUTTypeRTF;
}
NSString* availableType = [globalDragPboard availableTypeFromArray:[NSArray arrayWithObjects:(id)type, nil]];
if (availableType && IsValidType(availableType, allowFileURL)) {
diff --git a/widget/cocoa/nsScreenCocoa.mm b/widget/cocoa/nsScreenCocoa.mm
index 5a9ccb6e26c..1b72bbf82e1 100644
--- a/widget/cocoa/nsScreenCocoa.mm
+++ b/widget/cocoa/nsScreenCocoa.mm
@@ -44,7 +44,8 @@ nsScreenCocoa::GetRect(int32_t *outX, int32_t *outY, int32_t *outWidth, int32_t
{
NSRect frame = [mScreen frame];
- nsIntRect r = nsCocoaUtils::CocoaRectToGeckoRectDevPix(frame, BackingScaleFactor());
+ LayoutDeviceIntRect r =
+ nsCocoaUtils::CocoaRectToGeckoRectDevPix(frame, BackingScaleFactor());
*outX = r.x;
*outY = r.y;
@@ -59,7 +60,8 @@ nsScreenCocoa::GetAvailRect(int32_t *outX, int32_t *outY, int32_t *outWidth, int
{
NSRect frame = [mScreen visibleFrame];
- nsIntRect r = nsCocoaUtils::CocoaRectToGeckoRectDevPix(frame, BackingScaleFactor());
+ LayoutDeviceIntRect r =
+ nsCocoaUtils::CocoaRectToGeckoRectDevPix(frame, BackingScaleFactor());
*outX = r.x;
*outY = r.y;
diff --git a/widget/gtk/nsWindow.cpp b/widget/gtk/nsWindow.cpp
index 991c03a1b3f..81863428865 100644
--- a/widget/gtk/nsWindow.cpp
+++ b/widget/gtk/nsWindow.cpp
@@ -2233,8 +2233,7 @@ nsWindow::OnExposeEvent(cairo_t *cr)
// support for non-Thebes graphics, UpdateTranslucentWindowAlpha will be
// our private interface so we can rework things to avoid this.
boundsRect = region.GetBounds();
- dt->PushClipRect(Rect(boundsRect.x, boundsRect.y,
- boundsRect.width, boundsRect.height));
+ dt->PushClipRect(Rect(boundsRect));
} else {
gfxUtils::ClipToRegion(dt, region);
}
@@ -2244,18 +2243,19 @@ nsWindow::OnExposeEvent(cairo_t *cr)
// The double buffering is done here to extract the shape mask.
// (The shape mask won't be necessary when a visual with an alpha
// channel is used on compositing window managers.)
- layerBuffering = mozilla::layers::BufferMode::BUFFER_NONE;
- RefPtr destDT = dt->CreateSimilarDrawTarget(IntSize(boundsRect.width, boundsRect.height), SurfaceFormat::B8G8R8A8);
- ctx = new gfxContext(destDT, Point(boundsRect.x, boundsRect.y));
-#ifdef MOZ_HAVE_SHMIMAGE
+ layerBuffering = BufferMode::BUFFER_NONE;
+ RefPtr destDT = dt->CreateSimilarDrawTarget(boundsRect.Size(), SurfaceFormat::B8G8R8A8);
+ ctx = new gfxContext(destDT, boundsRect.TopLeft());
} else {
+#ifdef MOZ_HAVE_SHMIMAGE
if (nsShmImage::UseShm()) {
// We're using an xshm mapping as a back buffer.
- layerBuffering = mozilla::layers::BufferMode::BUFFER_NONE;
+ layerBuffering = BufferMode::BUFFER_NONE;
+ } else
#endif // MOZ_HAVE_SHMIMAGE
- } else {
+ {
// Get the layer manager to do double buffering (if necessary).
- layerBuffering = mozilla::layers::BufferMode::BUFFERED;
+ layerBuffering = BufferMode::BUFFERED;
}
ctx = new gfxContext(dt);
}
@@ -2291,8 +2291,7 @@ nsWindow::OnExposeEvent(cairo_t *cr)
UpdateAlpha(surf, boundsRect);
- dt->DrawSurface(surf, Rect(boundsRect.x, boundsRect.y, boundsRect.width, boundsRect.height),
- Rect(0, 0, boundsRect.width, boundsRect.height),
+ dt->DrawSurface(surf, Rect(boundsRect), Rect(0, 0, boundsRect.width, boundsRect.height),
DrawSurfaceOptions(Filter::POINT), DrawOptions(1.0f, CompositionOp::OP_SOURCE));
}
}
@@ -2346,7 +2345,7 @@ nsWindow::UpdateAlpha(SourceSurface* aSourceSurface, nsIntRect aBoundsRect)
stride, SurfaceFormat::A8);
if (drawTarget) {
- drawTarget->DrawSurface(aSourceSurface, Rect(aBoundsRect.x, aBoundsRect.y, aBoundsRect.width, aBoundsRect.height),
+ drawTarget->DrawSurface(aSourceSurface, Rect(0, 0, aBoundsRect.width, aBoundsRect.height),
Rect(0, 0, aSourceSurface->GetSize().width, aSourceSurface->GetSize().height),
DrawSurfaceOptions(Filter::POINT), DrawOptions(1.0f, CompositionOp::OP_SOURCE));
}
diff --git a/widget/nsBaseWidget.cpp b/widget/nsBaseWidget.cpp
index 1eb406e6ff6..31b40108403 100644
--- a/widget/nsBaseWidget.cpp
+++ b/widget/nsBaseWidget.cpp
@@ -1479,7 +1479,7 @@ nsBaseWidget::SetWindowTitlebarColor(nscolor aColor, bool aActive)
}
bool
-nsBaseWidget::ShowsResizeIndicator(nsIntRect* aResizerRect)
+nsBaseWidget::ShowsResizeIndicator(LayoutDeviceIntRect* aResizerRect)
{
return false;
}
diff --git a/widget/nsBaseWidget.h b/widget/nsBaseWidget.h
index bc179660f30..51673e3382f 100644
--- a/widget/nsBaseWidget.h
+++ b/widget/nsBaseWidget.h
@@ -167,7 +167,7 @@ public:
virtual bool PreRender(LayerManagerComposite* aManager) override { return true; }
virtual void PostRender(LayerManagerComposite* aManager) override {}
virtual void DrawWindowUnderlay(LayerManagerComposite* aManager, nsIntRect aRect) override {}
- virtual void DrawWindowOverlay(LayerManagerComposite* aManager, nsIntRect aRect) override {}
+ virtual void DrawWindowOverlay(LayerManagerComposite* aManager, LayoutDeviceIntRect aRect) override {}
virtual already_AddRefed StartRemoteDrawing() override;
virtual void EndRemoteDrawing() override { };
virtual void CleanupRemoteDrawing() override { };
@@ -202,7 +202,7 @@ public:
NS_IMETHOD SetIcon(const nsAString &anIconSpec) override;
NS_IMETHOD SetWindowTitlebarColor(nscolor aColor, bool aActive) override;
virtual void SetDrawsInTitlebar(bool aState) override {}
- virtual bool ShowsResizeIndicator(nsIntRect* aResizerRect) override;
+ virtual bool ShowsResizeIndicator(LayoutDeviceIntRect* aResizerRect) override;
virtual void FreeNativeData(void * data, uint32_t aDataType) override {}
NS_IMETHOD BeginResizeDrag(mozilla::WidgetGUIEvent* aEvent,
int32_t aHorizontal,
diff --git a/widget/nsClipboardProxy.cpp b/widget/nsClipboardProxy.cpp
index b88fd1d637f..28f05ae97d2 100644
--- a/widget/nsClipboardProxy.cpp
+++ b/widget/nsClipboardProxy.cpp
@@ -92,7 +92,8 @@ nsClipboardProxy::GetData(nsITransferable *aTransferable, int32_t aWhichClipboar
rv = aTransferable->SetTransferData(flavor.get(), stream, sizeof(nsISupports*));
NS_ENSURE_SUCCESS(rv, rv);
- } else if (flavor.EqualsLiteral(kNativeHTMLMime)) {
+ } else if (flavor.EqualsLiteral(kNativeHTMLMime) ||
+ flavor.EqualsLiteral(kRTFMime)) {
nsCOMPtr dataWrapper =
do_CreateInstance(NS_SUPPORTS_CSTRING_CONTRACTID, &rv);
NS_ENSURE_SUCCESS(rv, rv);
diff --git a/widget/nsITransferable.idl b/widget/nsITransferable.idl
index bc19b1b5ab0..ab35d69f973 100644
--- a/widget/nsITransferable.idl
+++ b/widget/nsITransferable.idl
@@ -15,6 +15,7 @@ interface nsIDOMNode;
// these probably shouldn't live here, but in some central repository shared
// by the entire app.
#define kTextMime "text/plain"
+#define kRTFMime "text/rtf"
#define kUnicodeMime "text/unicode"
#define kMozTextInternal "text/x-moz-text-internal" // text data which isn't suppoed to be parsed by other apps.
#define kHTMLMime "text/html"
diff --git a/widget/nsIWidget.h b/widget/nsIWidget.h
index 0584e6c96ad..0c3b36c72f8 100644
--- a/widget/nsIWidget.h
+++ b/widget/nsIWidget.h
@@ -335,6 +335,7 @@ class nsIWidget : public nsISupports {
typedef mozilla::LayoutDeviceIntMargin LayoutDeviceIntMargin;
typedef mozilla::LayoutDeviceIntPoint LayoutDeviceIntPoint;
typedef mozilla::LayoutDeviceIntRect LayoutDeviceIntRect;
+ typedef mozilla::LayoutDeviceIntRegion LayoutDeviceIntRegion;
typedef mozilla::LayoutDeviceIntSize LayoutDeviceIntSize;
// Used in UpdateThemeGeometries.
@@ -1258,7 +1259,8 @@ class nsIWidget : public nsISupports {
*
* Always called from the compositing thread.
*/
- virtual void DrawWindowOverlay(LayerManagerComposite* aManager, nsIntRect aRect) = 0;
+ virtual void DrawWindowOverlay(LayerManagerComposite* aManager,
+ LayoutDeviceIntRect aRect) = 0;
/**
* Return a DrawTarget for the window which can be composited into.
@@ -1500,7 +1502,7 @@ class nsIWidget : public nsISupports {
* @param aResizerRect The resizer's rect in device pixels.
* @return Whether a resize widget is shown.
*/
- virtual bool ShowsResizeIndicator(nsIntRect* aResizerRect) = 0;
+ virtual bool ShowsResizeIndicator(LayoutDeviceIntRect* aResizerRect) = 0;
/**
* Return the popup that was last rolled up, or null if there isn't one.
diff --git a/widget/nsPrimitiveHelpers.cpp b/widget/nsPrimitiveHelpers.cpp
index 71c88ca37e1..fe70262061c 100644
--- a/widget/nsPrimitiveHelpers.cpp
+++ b/widget/nsPrimitiveHelpers.cpp
@@ -45,7 +45,8 @@ nsPrimitiveHelpers :: CreatePrimitiveForData ( const char* aFlavor, const void*
if ( !aPrimitive )
return;
- if ( strcmp(aFlavor,kTextMime) == 0 || strcmp(aFlavor,kNativeHTMLMime) == 0 ) {
+ if ( strcmp(aFlavor,kTextMime) == 0 || strcmp(aFlavor,kNativeHTMLMime) == 0 ||
+ strcmp(aFlavor,kRTFMime) == 0) {
nsCOMPtr primitive =
do_CreateInstance(NS_SUPPORTS_CSTRING_CONTRACTID);
if ( primitive ) {
@@ -79,6 +80,40 @@ nsPrimitiveHelpers :: CreatePrimitiveForData ( const char* aFlavor, const void*
} // CreatePrimitiveForData
+//
+// CreatePrimitiveForCFHTML
+//
+// Platform specific CreatePrimitive, windows CF_HTML.
+//
+void
+nsPrimitiveHelpers :: CreatePrimitiveForCFHTML ( const void* aDataBuff,
+ uint32_t* aDataLen, nsISupports** aPrimitive )
+{
+ if (!aPrimitive)
+ return;
+
+ nsCOMPtr primitive =
+ do_CreateInstance(NS_SUPPORTS_STRING_CONTRACTID);
+ if (!primitive)
+ return;
+
+ // We need to duplicate the input buffer, since the removal of linebreaks
+ // might reallocte it.
+ void* utf8 = moz_xmalloc(*aDataLen);
+ if (!utf8)
+ return;
+ memcpy(utf8, aDataBuff, *aDataLen);
+ int32_t signedLen = static_cast(*aDataLen);
+ nsLinebreakHelpers::ConvertPlatformToDOMLinebreaks(kTextMime, &utf8, &signedLen);
+ *aDataLen = signedLen;
+
+ nsAutoString str(NS_ConvertUTF8toUTF16(reinterpret_cast(utf8), *aDataLen));
+ free(utf8);
+ *aDataLen = str.Length() * sizeof(char16_t);
+ primitive->SetData(str);
+ NS_ADDREF(*aPrimitive = primitive);
+}
+
//
// CreateDataFromPrimitive
@@ -136,7 +171,7 @@ nsLinebreakHelpers :: ConvertPlatformToDOMLinebreaks ( const char* inFlavor, voi
nsresult retVal = NS_OK;
- if ( strcmp(inFlavor, "text/plain") == 0 ) {
+ if ( strcmp(inFlavor, kTextMime) == 0 || strcmp(inFlavor, kRTFMime) == 0) {
char* buffAsChars = reinterpret_cast(*ioData);
char* oldBuffer = buffAsChars;
retVal = nsLinebreakConverter::ConvertLineBreaksInSitu ( &buffAsChars, nsLinebreakConverter::eLinebreakAny,
diff --git a/widget/nsPrimitiveHelpers.h b/widget/nsPrimitiveHelpers.h
index d6a9dae5319..952803a6606 100644
--- a/widget/nsPrimitiveHelpers.h
+++ b/widget/nsPrimitiveHelpers.h
@@ -23,6 +23,10 @@ public:
static void CreatePrimitiveForData ( const char* aFlavor, const void* aDataBuff,
uint32_t aDataLen, nsISupports** aPrimitive ) ;
+ // A specific case of CreatePrimitive for windows CF_HTML handling in DataTransfer
+ static void CreatePrimitiveForCFHTML ( const void* aDataBuff,
+ uint32_t* aDataLen, nsISupports** aPrimitive ) ;
+
// Given a nsISupports* primitive and the flavor it represents, creates a new data
// buffer with the data in it. This data will be null terminated, but the length
// parameter does not reflect that.
diff --git a/widget/windows/nsClipboard.cpp b/widget/windows/nsClipboard.cpp
index 32190a29f36..67cdee82ea8 100644
--- a/widget/windows/nsClipboard.cpp
+++ b/widget/windows/nsClipboard.cpp
@@ -95,6 +95,8 @@ UINT nsClipboard::GetFormat(const char* aMimeStr)
if (strcmp(aMimeStr, kTextMime) == 0 ||
strcmp(aMimeStr, kUnicodeMime) == 0)
format = CF_UNICODETEXT;
+ else if (strcmp(aMimeStr, kRTFMime) == 0)
+ format = ::RegisterClipboardFormat(L"Rich Text Format");
else if (strcmp(aMimeStr, kJPEGImageMime) == 0 ||
strcmp(aMimeStr, kJPGImageMime) == 0 ||
strcmp(aMimeStr, kPNGImageMime) == 0)
@@ -102,7 +104,8 @@ UINT nsClipboard::GetFormat(const char* aMimeStr)
else if (strcmp(aMimeStr, kFileMime) == 0 ||
strcmp(aMimeStr, kFilePromiseMime) == 0)
format = CF_HDROP;
- else if (strcmp(aMimeStr, kNativeHTMLMime) == 0)
+ else if (strcmp(aMimeStr, kNativeHTMLMime) == 0 ||
+ strcmp(aMimeStr, kHTMLMime) == 0)
format = CF_HTML;
else
format = ::RegisterClipboardFormatW(NS_ConvertASCIItoUTF16(aMimeStr).get());
@@ -624,11 +627,12 @@ nsresult nsClipboard::GetDataFromDataObject(IDataObject * aDataObject,
genericDataWrapper = do_QueryInterface(file);
free(data);
}
- else if ( strcmp(flavorStr, kNativeHTMLMime) == 0) {
+ else if ( strcmp(flavorStr, kNativeHTMLMime) == 0 ) {
+ uint32_t dummy;
// the editor folks want CF_HTML exactly as it's on the clipboard, no conversions,
// no fancy stuff. Pull it off the clipboard, stuff it into a wrapper and hand
// it back to them.
- if ( FindPlatformHTML(aDataObject, anIndex, &data, &dataLen) )
+ if ( FindPlatformHTML(aDataObject, anIndex, &data, &dummy, &dataLen) )
nsPrimitiveHelpers::CreatePrimitiveForData ( flavorStr, data, dataLen, getter_AddRefs(genericDataWrapper) );
else
{
@@ -637,6 +641,23 @@ nsresult nsClipboard::GetDataFromDataObject(IDataObject * aDataObject,
}
free(data);
}
+ else if ( strcmp(flavorStr, kHTMLMime) == 0 ) {
+ uint32_t startOfData = 0;
+ // The JS folks want CF_HTML exactly as it is on the clipboard, but
+ // minus the CF_HTML header index information.
+ // It also needs to be converted to UTF16 and have linebreaks changed.
+ if ( FindPlatformHTML(aDataObject, anIndex, &data, &startOfData, &dataLen) ) {
+ dataLen -= startOfData;
+ nsPrimitiveHelpers::CreatePrimitiveForCFHTML ( static_cast(data) + startOfData,
+ &dataLen, getter_AddRefs(genericDataWrapper) );
+ }
+ else
+ {
+ free(data);
+ continue; // something wrong with this flavor, keep looking for other data
+ }
+ free(data);
+ }
else if ( strcmp(flavorStr, kJPEGImageMime) == 0 ||
strcmp(flavorStr, kJPGImageMime) == 0 ||
strcmp(flavorStr, kPNGImageMime) == 0) {
@@ -651,6 +672,12 @@ nsresult nsClipboard::GetDataFromDataObject(IDataObject * aDataObject,
nsLinebreakHelpers::ConvertPlatformToDOMLinebreaks ( flavorStr, &data, &signedLen );
dataLen = signedLen;
+ if (strcmp(flavorStr, kRTFMime) == 0) {
+ // RTF on Windows is known to sometimes deliver an extra null byte.
+ if (dataLen > 0 && static_cast(data)[dataLen - 1] == '\0')
+ dataLen--;
+ }
+
nsPrimitiveHelpers::CreatePrimitiveForData ( flavorStr, data, dataLen, getter_AddRefs(genericDataWrapper) );
free(data);
}
@@ -678,7 +705,9 @@ nsresult nsClipboard::GetDataFromDataObject(IDataObject * aDataObject,
// Someone asked for the OS CF_HTML flavor. We give it back to them exactly as-is.
//
bool
-nsClipboard :: FindPlatformHTML ( IDataObject* inDataObject, UINT inIndex, void** outData, uint32_t* outDataLen )
+nsClipboard :: FindPlatformHTML ( IDataObject* inDataObject, UINT inIndex,
+ void** outData, uint32_t* outStartOfData,
+ uint32_t* outDataLen )
{
// Reference: MSDN doc entitled "HTML Clipboard Format"
// http://msdn.microsoft.com/en-us/library/aa767917(VS.85).aspx#unknown_854
@@ -721,6 +750,10 @@ nsClipboard :: FindPlatformHTML ( IDataObject* inDataObject, UINT inIndex, void*
// We want to return the buffer not offset by startOfData because it will be
// parsed out later (probably by nsHTMLEditor::ParseCFHTML) when it is still
// in CF_HTML format.
+
+ // We return the byte offset from the start of the data buffer to where the
+ // HTML data starts. The caller might want to extract the HTML only.
+ *outStartOfData = startOfData;
*outDataLen = endOfData;
return true;
}
diff --git a/widget/windows/nsClipboard.h b/widget/windows/nsClipboard.h
index b1157eb510e..39456ce3e87 100644
--- a/widget/windows/nsClipboard.h
+++ b/widget/windows/nsClipboard.h
@@ -63,7 +63,8 @@ protected:
static bool FindURLFromLocalFile ( IDataObject* inDataObject, UINT inIndex, void** outData, uint32_t* outDataLen ) ;
static bool FindURLFromNativeURL ( IDataObject* inDataObject, UINT inIndex, void** outData, uint32_t* outDataLen ) ;
static bool FindUnicodeFromPlainText ( IDataObject* inDataObject, UINT inIndex, void** outData, uint32_t* outDataLen ) ;
- static bool FindPlatformHTML ( IDataObject* inDataObject, UINT inIndex, void** outData, uint32_t* outDataLen );
+ static bool FindPlatformHTML ( IDataObject* inDataObject, UINT inIndex, void** outData,
+ uint32_t* outStartOfData, uint32_t* outDataLen );
static void ResolveShortcut ( nsIFile* inFileName, nsACString& outURL ) ;
nsIWidget * mWindow;
diff --git a/widget/windows/nsDragService.cpp b/widget/windows/nsDragService.cpp
index 6d4a30988f1..3462ee6cf3e 100644
--- a/widget/windows/nsDragService.cpp
+++ b/widget/windows/nsDragService.cpp
@@ -539,6 +539,14 @@ nsDragService::IsDataFlavorSupported(const char *aDataFlavor, bool *_retval)
if (mDataObject->QueryGetData(&fe) == S_OK)
*_retval = true; // found it!
}
+ else if (strcmp(aDataFlavor, kHTMLMime) == 0) {
+ // If the client wants html, maybe it's in "HTML Format".
+ format = nsClipboard::GetFormat(kHTMLMime);
+ SET_FORMATETC(fe, format, 0, DVASPECT_CONTENT, -1,
+ TYMED_HGLOBAL);
+ if (mDataObject->QueryGetData(&fe) == S_OK)
+ *_retval = true; // found it!
+ }
} // else try again
}
diff --git a/xpcom/base/nsCycleCollector.cpp b/xpcom/base/nsCycleCollector.cpp
index 2f4572b5175..a046f6ea705 100644
--- a/xpcom/base/nsCycleCollector.cpp
+++ b/xpcom/base/nsCycleCollector.cpp
@@ -2832,7 +2832,7 @@ nsCycleCollector::ForgetSkippable(bool aRemoveChildlessNodes,
mozilla::Maybe marker;
if (NS_IsMainThread()) {
- marker.emplace("nsCycleCollector::ForgetSkippable");
+ marker.emplace("nsCycleCollector::ForgetSkippable", MarkerStackRequest::NO_STACK);
}
// If we remove things from the purple buffer during graph building, we may
@@ -3584,7 +3584,7 @@ nsCycleCollector::Collect(ccType aCCType,
mozilla::Maybe marker;
if (NS_IsMainThread()) {
- marker.emplace("nsCycleCollector::Collect");
+ marker.emplace("nsCycleCollector::Collect", MarkerStackRequest::NO_STACK);
}
bool startedIdle = IsIdle();