зеркало из https://github.com/mozilla/gecko-dev.git
Merge m-c to autoland, a=merge
MozReview-Commit-ID: 6wWFiDQ0QJ8
This commit is contained in:
Коммит
269e3163e5
|
@ -125,7 +125,7 @@ const addView = curry((options, {document, window}) => {
|
|||
toolbar.setAttribute("id", "inner-" + options.id);
|
||||
toolbar.setAttribute("defaultset", options.items.join(","));
|
||||
toolbar.setAttribute("customizable", "true");
|
||||
toolbar.setAttribute("style", "-moz-appearance: none; overflow: hidden");
|
||||
toolbar.setAttribute("style", "-moz-appearance: none; overflow: hidden; border: 0;");
|
||||
toolbar.setAttribute("mode", "icons");
|
||||
toolbar.setAttribute("iconsize", "small");
|
||||
toolbar.setAttribute("context", "toolbar-context-menu");
|
||||
|
|
|
@ -2,7 +2,7 @@
|
|||
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
||||
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
||||
|
||||
/* global XPCOMUtils, Services, PlacesUtils, gPrincipal, EventEmitter */
|
||||
/* global XPCOMUtils, Services, PlacesUtils, EventEmitter */
|
||||
/* global gLinks */
|
||||
/* exported PlacesProvider */
|
||||
|
||||
|
@ -22,52 +22,14 @@ XPCOMUtils.defineLazyGetter(this, "EventEmitter", function() {
|
|||
return EventEmitter;
|
||||
});
|
||||
|
||||
XPCOMUtils.defineLazyGetter(this, "gPrincipal", function() {
|
||||
let uri = Services.io.newURI("about:newtab");
|
||||
return Services.scriptSecurityManager.getNoAppCodebasePrincipal(uri);
|
||||
});
|
||||
|
||||
XPCOMUtils.defineLazyModuleGetter(this, "Task",
|
||||
"resource://gre/modules/Task.jsm");
|
||||
XPCOMUtils.defineLazyModuleGetter(this, "NewTabUtils",
|
||||
"resource://gre/modules/NewTabUtils.jsm");
|
||||
|
||||
// The maximum number of results PlacesProvider retrieves from history.
|
||||
const HISTORY_RESULTS_LIMIT = 100;
|
||||
|
||||
/**
|
||||
* Singleton that checks if a given link should be displayed on about:newtab
|
||||
* or if we should rather not do it for security reasons. URIs that inherit
|
||||
* their caller's principal will be filtered.
|
||||
*/
|
||||
let LinkChecker = {
|
||||
_cache: new Map(),
|
||||
|
||||
get flags() {
|
||||
return Ci.nsIScriptSecurityManager.DISALLOW_INHERIT_PRINCIPAL |
|
||||
Ci.nsIScriptSecurityManager.DONT_REPORT_ERRORS;
|
||||
},
|
||||
|
||||
checkLoadURI: function LinkChecker_checkLoadURI(aURI) {
|
||||
if (!this._cache.has(aURI)) {
|
||||
this._cache.set(aURI, this._doCheckLoadURI(aURI));
|
||||
}
|
||||
|
||||
return this._cache.get(aURI);
|
||||
},
|
||||
|
||||
_doCheckLoadURI: function LinkChecker_doCheckLoadURI(aURI) {
|
||||
let result = false;
|
||||
try {
|
||||
Services.scriptSecurityManager.
|
||||
checkLoadURIStrWithPrincipal(gPrincipal, aURI, this.flags);
|
||||
result = true;
|
||||
} catch (e) {
|
||||
// We got a weird URI or one that would inherit the caller's principal.
|
||||
Cu.reportError(e);
|
||||
}
|
||||
return result;
|
||||
}
|
||||
};
|
||||
|
||||
/* Queries history to retrieve the most visited sites. Emits events when the
|
||||
* history changes.
|
||||
* Implements the EventEmitter interface.
|
||||
|
@ -105,7 +67,8 @@ Links.prototype = {
|
|||
aNewFrecency, aGUID, aHidden, aLastVisitDate) { // jshint ignore:line
|
||||
// The implementation of the query in getLinks excludes hidden and
|
||||
// unvisited pages, so it's important to exclude them here, too.
|
||||
if (!aHidden && aLastVisitDate) {
|
||||
if (!aHidden && aLastVisitDate &&
|
||||
NewTabUtils.linkChecker.checkLoadURI(aURI.spec)) {
|
||||
gLinks.emit("linkChanged", {
|
||||
url: aURI.spec,
|
||||
frecency: aNewFrecency,
|
||||
|
@ -122,10 +85,12 @@ Links.prototype = {
|
|||
},
|
||||
|
||||
onTitleChanged: function historyObserver_onTitleChanged(aURI, aNewTitle) {
|
||||
if (NewTabUtils.linkChecker.checkLoadURI(aURI.spec)) {
|
||||
gLinks.emit("linkChanged", {
|
||||
url: aURI.spec,
|
||||
title: aNewTitle
|
||||
});
|
||||
}
|
||||
},
|
||||
|
||||
QueryInterface: XPCOMUtils.generateQI([Ci.nsINavHistoryObserver,
|
||||
|
@ -184,7 +149,7 @@ Links.prototype = {
|
|||
params: {limit: this.maxNumLinks}
|
||||
});
|
||||
|
||||
return links.filter(link => LinkChecker.checkLoadURI(link.url));
|
||||
return links.filter(link => NewTabUtils.linkChecker.checkLoadURI(link.url));
|
||||
}),
|
||||
|
||||
/**
|
||||
|
@ -247,6 +212,10 @@ Links.prototype = {
|
|||
const gLinks = new Links(); // jshint ignore:line
|
||||
|
||||
let PlacesProvider = {
|
||||
LinkChecker,
|
||||
links: gLinks,
|
||||
};
|
||||
|
||||
// Kept only for backwards-compatibility
|
||||
XPCOMUtils.defineLazyGetter(PlacesProvider, "LinkChecker",
|
||||
() => NewTabUtils.linkChecker);
|
||||
|
||||
|
|
|
@ -59,13 +59,13 @@ function makeVisit(index, daysAgo, isTyped, domain = TEST_URL) {
|
|||
add_task(function test_LinkChecker_securityCheck() {
|
||||
|
||||
let urls = [
|
||||
{url: "file://home/file/image.png", expected: false},
|
||||
{url: "resource:///modules/PlacesProvider.jsm", expected: false},
|
||||
{url: "javascript:alert('hello')", expected: false}, // jshint ignore:line
|
||||
{url: "data:image/png;base64,XXX", expected: false},
|
||||
{url: "about:newtab", expected: true},
|
||||
{url: "https://example.com", expected: true},
|
||||
{url: "ftp://example.com", expected: true},
|
||||
{url: "file://home/file/image.png", expected: true},
|
||||
{url: "resource:///modules/PlacesProvider.jsm", expected: true},
|
||||
];
|
||||
for (let {url, expected} of urls) {
|
||||
let observed = PlacesProvider.LinkChecker.checkLoadURI(url);
|
||||
|
|
|
@ -37,5 +37,13 @@
|
|||
"algorithm": "sha512",
|
||||
"filename": "clang.tar.bz2",
|
||||
"unpack": true
|
||||
},
|
||||
{
|
||||
"algorithm": "sha512",
|
||||
"visibility": "public",
|
||||
"filename": "makecab.tar.bz2",
|
||||
"unpack": true,
|
||||
"digest": "da1f7685e5bc49a5ffbe5b4a3678ac7a58a0e125031726d30e2bacbbffa53d6efb9fd9f00d6dff5b54cff9412a103efa564c2af6f8fccc63082f6939181769f8",
|
||||
"size": 296777
|
||||
}
|
||||
]
|
||||
|
|
|
@ -29,5 +29,13 @@
|
|||
"algorithm": "sha512",
|
||||
"filename": "vs2015u3.zip",
|
||||
"unpack": true
|
||||
},
|
||||
{
|
||||
"algorithm": "sha512",
|
||||
"visibility": "public",
|
||||
"filename": "makecab.tar.bz2",
|
||||
"unpack": true,
|
||||
"digest": "da1f7685e5bc49a5ffbe5b4a3678ac7a58a0e125031726d30e2bacbbffa53d6efb9fd9f00d6dff5b54cff9412a103efa564c2af6f8fccc63082f6939181769f8",
|
||||
"size": 296777
|
||||
}
|
||||
]
|
||||
|
|
|
@ -38,5 +38,13 @@
|
|||
"algorithm": "sha512",
|
||||
"filename": "clang.tar.bz2",
|
||||
"unpack": true
|
||||
},
|
||||
{
|
||||
"algorithm": "sha512",
|
||||
"visibility": "public",
|
||||
"filename": "makecab.tar.bz2",
|
||||
"unpack": true,
|
||||
"digest": "da1f7685e5bc49a5ffbe5b4a3678ac7a58a0e125031726d30e2bacbbffa53d6efb9fd9f00d6dff5b54cff9412a103efa564c2af6f8fccc63082f6939181769f8",
|
||||
"size": 296777
|
||||
}
|
||||
]
|
||||
|
|
|
@ -30,5 +30,13 @@
|
|||
"algorithm": "sha512",
|
||||
"filename": "vs2015u3.zip",
|
||||
"unpack": true
|
||||
},
|
||||
{
|
||||
"algorithm": "sha512",
|
||||
"visibility": "public",
|
||||
"filename": "makecab.tar.bz2",
|
||||
"unpack": true,
|
||||
"digest": "da1f7685e5bc49a5ffbe5b4a3678ac7a58a0e125031726d30e2bacbbffa53d6efb9fd9f00d6dff5b54cff9412a103efa564c2af6f8fccc63082f6939181769f8",
|
||||
"size": 296777
|
||||
}
|
||||
]
|
||||
|
|
|
@ -206,9 +206,12 @@ body:not(.compact) .newtab-site[type=sponsored] .newtab-thumbnail {
|
|||
}
|
||||
|
||||
body.compact .newtab-title {
|
||||
background-color: hsla(0,0%,100%,.8);
|
||||
background-color: hsla(0,0%,100%,.85);
|
||||
font-size: 12px;
|
||||
line-height: 21px;
|
||||
border: 1px solid hsla(0,0%,80%,.8);
|
||||
border-top-color: hsla(0,0%,0%,.1);
|
||||
background-clip: padding-box;
|
||||
}
|
||||
|
||||
.newtab-title,
|
||||
|
@ -221,12 +224,6 @@ body.compact .newtab-suggested {
|
|||
color: black;
|
||||
}
|
||||
|
||||
body.compact .newtab-title {
|
||||
border: 1px solid hsla(0,0%,80%,.8);
|
||||
border-top-color: hsla(0,0%,0%,.1);
|
||||
background-clip: padding-box;
|
||||
}
|
||||
|
||||
.newtab-suggested[active] {
|
||||
background-color: rgba(51, 51, 51, 0.95);
|
||||
border: 0;
|
||||
|
@ -242,7 +239,7 @@ body:not(.compact) .newtab-site:hover .newtab-title {
|
|||
|
||||
body.compact .newtab-site:hover .newtab-title {
|
||||
color: white;
|
||||
background-color: hsla(0,0%,20%,.8);
|
||||
background-color: hsla(0,0%,20%,.85);
|
||||
border-color: hsla(0,0%,0%,.8);
|
||||
border-top-color: white;
|
||||
}
|
||||
|
|
|
@ -69,7 +69,7 @@ if test -n "$USE_ICU"; then
|
|||
fi
|
||||
fi
|
||||
|
||||
version=`sed -n 's/^[[:space:]]*#[[:space:]]*define[[:space:]][[:space:]]*U_ICU_VERSION_MAJOR_NUM[[:space:]][[:space:]]*\([0-9][0-9]*\)[[:space:]]*$/\1/p' "$icudir/common/unicode/uvernum.h"`
|
||||
version=`sed -n 's/^[[[:space:]]]*#[[:space:]]*define[[:space:]][[:space:]]*U_ICU_VERSION_MAJOR_NUM[[:space:]][[:space:]]*\([0-9][0-9]*\)[[:space:]]*$/\1/p' "$icudir/common/unicode/uvernum.h"`
|
||||
if test x"$version" = x; then
|
||||
AC_MSG_ERROR([cannot determine icu version number from uvernum.h header file $lineno])
|
||||
fi
|
||||
|
|
|
@ -416,3 +416,6 @@ def alter_path(sdk_bin_path):
|
|||
return path
|
||||
|
||||
set_config('PATH', alter_path)
|
||||
|
||||
check_prog('MAKECAB', ('makecab.exe',))
|
||||
|
||||
|
|
|
@ -14,3 +14,4 @@ fi
|
|||
MOZ_AUTOMATION_INSTALLER=${MOZ_AUTOMATION_INSTALLER-1}
|
||||
|
||||
export SOCORRO_SYMBOL_UPLOAD_TOKEN_FILE=c:/builds/crash-stats-api.token
|
||||
export MAKECAB=$topsrcdir/makecab.exe
|
||||
|
|
|
@ -61,3 +61,7 @@ if CONFIG['ENABLE_TESTS']:
|
|||
include('/ipc/chromium/chromium-config.mozbuild')
|
||||
|
||||
FINAL_LIBRARY = 'xul'
|
||||
|
||||
with Files("**"):
|
||||
BUG_COMPONENT = ("Core", "Security: CAPS")
|
||||
|
||||
|
|
|
@ -30,6 +30,8 @@ define(function (require, exports, module) {
|
|||
propTypes: {
|
||||
// @TODO Change this to Object.values once it's supported in Node's version of V8
|
||||
mode: React.PropTypes.oneOf(Object.keys(MODE).map(key => MODE[key])),
|
||||
objectLink: React.PropTypes.func,
|
||||
object: React.PropTypes.array.isRequired,
|
||||
},
|
||||
|
||||
getTitle: function (object, context) {
|
||||
|
@ -170,6 +172,12 @@ define(function (require, exports, module) {
|
|||
let ItemRep = React.createFactory(React.createClass({
|
||||
displayName: "ItemRep",
|
||||
|
||||
propTypes: {
|
||||
object: React.PropTypes.any.isRequired,
|
||||
delim: React.PropTypes.string.isRequired,
|
||||
mode: React.PropTypes.symbol,
|
||||
},
|
||||
|
||||
render: wrapRender(function () {
|
||||
const { Rep } = createFactories(require("./rep"));
|
||||
|
||||
|
|
|
@ -29,7 +29,8 @@ define(function (require, exports, module) {
|
|||
displayName: "Attr",
|
||||
|
||||
propTypes: {
|
||||
object: React.PropTypes.object.isRequired
|
||||
object: React.PropTypes.object.isRequired,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTitle: function (grip) {
|
||||
|
|
|
@ -21,6 +21,10 @@ define(function (require, exports, module) {
|
|||
const Caption = React.createClass({
|
||||
displayName: "Caption",
|
||||
|
||||
propTypes: {
|
||||
object: React.PropTypes.object,
|
||||
},
|
||||
|
||||
render: wrapRender(function () {
|
||||
return (
|
||||
DOM.span({"className": "caption"}, this.props.object)
|
||||
|
|
|
@ -26,7 +26,8 @@ define(function (require, exports, module) {
|
|||
displayName: "Date",
|
||||
|
||||
propTypes: {
|
||||
object: React.PropTypes.object.isRequired
|
||||
object: React.PropTypes.object.isRequired,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTitle: function (grip) {
|
||||
|
|
|
@ -27,7 +27,8 @@ define(function (require, exports, module) {
|
|||
displayName: "Document",
|
||||
|
||||
propTypes: {
|
||||
object: React.PropTypes.object.isRequired
|
||||
object: React.PropTypes.object.isRequired,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getLocation: function (grip) {
|
||||
|
|
|
@ -31,6 +31,9 @@ define(function (require, exports, module) {
|
|||
object: React.PropTypes.object.isRequired,
|
||||
// @TODO Change this to Object.values once it's supported in Node's version of V8
|
||||
mode: React.PropTypes.oneOf(Object.keys(MODE).map(key => MODE[key])),
|
||||
onDOMNodeMouseOver: React.PropTypes.func,
|
||||
onDOMNodeMouseOut: React.PropTypes.func,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getElements: function (grip, mode) {
|
||||
|
|
|
@ -26,6 +26,7 @@ define(function (require, exports, module) {
|
|||
object: React.PropTypes.object.isRequired,
|
||||
// @TODO Change this to Object.values once it's supported in Node's version of V8
|
||||
mode: React.PropTypes.oneOf(Object.keys(MODE).map(key => MODE[key])),
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
render: wrapRender(function () {
|
||||
|
|
|
@ -27,7 +27,8 @@ define(function (require, exports, module) {
|
|||
displayName: "Func",
|
||||
|
||||
propTypes: {
|
||||
object: React.PropTypes.object.isRequired
|
||||
object: React.PropTypes.object.isRequired,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTitle: function (grip) {
|
||||
|
|
|
@ -33,6 +33,7 @@ define(function (require, exports, module) {
|
|||
// @TODO Change this to Object.values once it's supported in Node's version of V8
|
||||
mode: React.PropTypes.oneOf(Object.keys(MODE).map(key => MODE[key])),
|
||||
provider: React.PropTypes.object,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getLength: function (grip) {
|
||||
|
|
|
@ -30,6 +30,8 @@ define(function (require, exports, module) {
|
|||
object: React.PropTypes.object,
|
||||
// @TODO Change this to Object.values once it's supported in Node's version of V8
|
||||
mode: React.PropTypes.oneOf(Object.keys(MODE).map(key => MODE[key])),
|
||||
objectLink: React.PropTypes.func,
|
||||
isInterestingEntry: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTitle: function (object) {
|
||||
|
|
|
@ -35,6 +35,7 @@ define(function (require, exports, module) {
|
|||
mode: React.PropTypes.oneOf(Object.keys(MODE).map(key => MODE[key])),
|
||||
isInterestingProp: React.PropTypes.func,
|
||||
title: React.PropTypes.string,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTitle: function (object) {
|
||||
|
|
|
@ -22,6 +22,10 @@ define(function (require, exports, module) {
|
|||
const InfinityRep = React.createClass({
|
||||
displayName: "Infinity",
|
||||
|
||||
propTypes: {
|
||||
object: React.PropTypes.object.isRequired,
|
||||
},
|
||||
|
||||
render: wrapRender(function () {
|
||||
return (
|
||||
span({className: "objectBox objectBox-number"},
|
||||
|
|
|
@ -25,6 +25,9 @@ define(function (require, exports, module) {
|
|||
propTypes: {
|
||||
useQuotes: React.PropTypes.bool,
|
||||
style: React.PropTypes.object,
|
||||
cropLimit: React.PropTypes.number.isRequired,
|
||||
member: React.PropTypes.string,
|
||||
object: React.PropTypes.object.isRequired,
|
||||
},
|
||||
|
||||
getDefaultProps: function () {
|
||||
|
|
|
@ -22,6 +22,13 @@ define(function (require, exports, module) {
|
|||
const Number = React.createClass({
|
||||
displayName: "Number",
|
||||
|
||||
propTypes: {
|
||||
object: React.PropTypes.oneOfType([
|
||||
React.PropTypes.object,
|
||||
React.PropTypes.number,
|
||||
]).isRequired
|
||||
},
|
||||
|
||||
stringify: function (object) {
|
||||
let isNegativeZero = Object.is(object, -0) ||
|
||||
(object.type && object.type == "-0");
|
||||
|
|
|
@ -27,6 +27,7 @@ define(function (require, exports, module) {
|
|||
|
||||
propTypes: {
|
||||
object: React.PropTypes.object.isRequired,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTitle: function (grip) {
|
||||
|
|
|
@ -28,6 +28,7 @@ define(function (require, exports, module) {
|
|||
|
||||
propTypes: {
|
||||
object: React.PropTypes.object.isRequired,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTitle: function (grip) {
|
||||
|
|
|
@ -29,6 +29,7 @@ define(function (require, exports, module) {
|
|||
object: React.PropTypes.object,
|
||||
// @TODO Change this to Object.values once it's supported in Node's version of V8
|
||||
mode: React.PropTypes.oneOf(Object.keys(MODE).map(key => MODE[key])),
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTitle: function (object) {
|
||||
|
|
|
@ -31,6 +31,7 @@ define(function (require, exports, module) {
|
|||
object: React.PropTypes.object.isRequired,
|
||||
// @TODO Change this to Object.values once it's supported in Node's version of V8
|
||||
mode: React.PropTypes.oneOf(Object.keys(MODE).map(key => MODE[key])),
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTitle: function (object) {
|
||||
|
|
|
@ -37,6 +37,7 @@ define(function (require, exports, module) {
|
|||
delim: React.PropTypes.string,
|
||||
// @TODO Change this to Object.values once it's supported in Node's version of V8
|
||||
mode: React.PropTypes.oneOf(Object.keys(MODE).map(key => MODE[key])),
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
render: wrapRender(function () {
|
||||
|
|
|
@ -27,6 +27,7 @@ define(function (require, exports, module) {
|
|||
|
||||
propTypes: {
|
||||
object: React.PropTypes.object.isRequired,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getSource: function (grip) {
|
||||
|
|
|
@ -28,6 +28,9 @@ define(function (require, exports, module) {
|
|||
propTypes: {
|
||||
useQuotes: React.PropTypes.bool,
|
||||
style: React.PropTypes.object,
|
||||
object: React.PropTypes.string.isRequired,
|
||||
member: React.PropTypes.any,
|
||||
cropLimit: React.PropTypes.number,
|
||||
},
|
||||
|
||||
getDefaultProps: function () {
|
||||
|
|
|
@ -28,6 +28,7 @@ define(function (require, exports, module) {
|
|||
|
||||
propTypes: {
|
||||
object: React.PropTypes.object.isRequired,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTitle: function (grip) {
|
||||
|
|
|
@ -31,6 +31,9 @@ define(function (require, exports, module) {
|
|||
object: React.PropTypes.object.isRequired,
|
||||
// @TODO Change this to Object.values once it's supported in Node's version of V8
|
||||
mode: React.PropTypes.oneOf(Object.keys(MODE).map(key => MODE[key])),
|
||||
objectLink: React.PropTypes.func,
|
||||
onDOMNodeMouseOver: React.PropTypes.func,
|
||||
onDOMNodeMouseOut: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTextContent: function (grip) {
|
||||
|
|
|
@ -28,6 +28,7 @@ define(function (require, exports, module) {
|
|||
|
||||
propTypes: {
|
||||
object: React.PropTypes.object.isRequired,
|
||||
objectLink: React.PropTypes.func,
|
||||
},
|
||||
|
||||
getTitle: function (grip) {
|
||||
|
|
|
@ -21,6 +21,10 @@ const SplitBox = React.createClass({
|
|||
className: PropTypes.string,
|
||||
// Initial size of controlled panel.
|
||||
initialSize: PropTypes.number,
|
||||
// Initial width of controlled panel.
|
||||
initialWidth: PropTypes.number,
|
||||
// Initial height of controlled panel.
|
||||
initialHeight: PropTypes.number,
|
||||
// Left/top panel
|
||||
startPanel: PropTypes.any,
|
||||
// Min panel size.
|
||||
|
@ -34,7 +38,9 @@ const SplitBox = React.createClass({
|
|||
// Size of the splitter handle bar.
|
||||
splitterSize: PropTypes.number,
|
||||
// True if the splitter bar is vertical (default is vertical).
|
||||
vert: PropTypes.bool
|
||||
vert: PropTypes.bool,
|
||||
// Style object.
|
||||
style: PropTypes.object,
|
||||
},
|
||||
|
||||
getDefaultProps() {
|
||||
|
|
|
@ -14,7 +14,7 @@ const l10n = new LocalizationHelper("devtools/client/locales/webconsole.properti
|
|||
const AsyncFrame = createFactory(createClass({
|
||||
displayName: "AsyncFrame",
|
||||
|
||||
PropTypes: {
|
||||
propTypes: {
|
||||
asyncCause: PropTypes.string.isRequired
|
||||
},
|
||||
|
||||
|
@ -31,10 +31,10 @@ const AsyncFrame = createFactory(createClass({
|
|||
const StackTrace = createClass({
|
||||
displayName: "StackTrace",
|
||||
|
||||
PropTypes: {
|
||||
propTypes: {
|
||||
stacktrace: PropTypes.array.isRequired,
|
||||
onViewSourceInDebugger: PropTypes.func.isRequired,
|
||||
onViewSourceInScratchpad: PropTypes.func.isRequired,
|
||||
onViewSourceInScratchpad: PropTypes.func,
|
||||
},
|
||||
|
||||
render() {
|
||||
|
|
|
@ -4,7 +4,8 @@
|
|||
/* eslint-env browser */
|
||||
"use strict";
|
||||
|
||||
const { DOM: dom, createClass, createFactory, PropTypes } = require("devtools/client/shared/vendor/react");
|
||||
const React = require("devtools/client/shared/vendor/react");
|
||||
const { DOM: dom, createClass, createFactory, PropTypes } = React;
|
||||
|
||||
const AUTO_EXPAND_DEPTH = 0;
|
||||
const NUMBER_OF_OFFSCREEN_ITEMS = 1;
|
||||
|
@ -630,6 +631,14 @@ module.exports = createClass({
|
|||
const ArrowExpander = createFactory(createClass({
|
||||
displayName: "ArrowExpander",
|
||||
|
||||
propTypes: {
|
||||
item: PropTypes.any.isRequired,
|
||||
visible: PropTypes.bool.isRequired,
|
||||
expanded: PropTypes.bool.isRequired,
|
||||
onCollapse: PropTypes.func.isRequired,
|
||||
onExpand: PropTypes.func.isRequired,
|
||||
},
|
||||
|
||||
shouldComponentUpdate(nextProps, nextState) {
|
||||
return this.props.item !== nextProps.item
|
||||
|| this.props.visible !== nextProps.visible
|
||||
|
@ -659,6 +668,23 @@ const ArrowExpander = createFactory(createClass({
|
|||
}));
|
||||
|
||||
const TreeNode = createFactory(createClass({
|
||||
propTypes: {
|
||||
focused: PropTypes.bool.isRequired,
|
||||
onFocusedNodeUnmount: PropTypes.func,
|
||||
item: PropTypes.any.isRequired,
|
||||
expanded: PropTypes.bool.isRequired,
|
||||
hasChildren: PropTypes.bool.isRequired,
|
||||
onExpand: PropTypes.func.isRequired,
|
||||
index: PropTypes.number.isRequired,
|
||||
first: PropTypes.bool,
|
||||
last: PropTypes.bool,
|
||||
onFocus: PropTypes.func,
|
||||
onBlur: PropTypes.func,
|
||||
onCollapse: PropTypes.func.isRequired,
|
||||
depth: PropTypes.number.isRequired,
|
||||
renderItem: PropTypes.func.isRequired,
|
||||
},
|
||||
|
||||
componentDidMount() {
|
||||
if (this.props.focused) {
|
||||
this.refs.button.focus();
|
||||
|
|
|
@ -21,9 +21,11 @@ add_task(function* () {
|
|||
dir.append(TEST_FILE);
|
||||
let uri = Services.io.newFileURI(dir);
|
||||
|
||||
// We need a file remote type to make sure we don't switch processes when we
|
||||
// load the file:// URI.
|
||||
let { browser } = yield loadTab("about:blank", E10SUtils.FILE_REMOTE_TYPE);
|
||||
// Open tab with correct remote type so we don't switch processes when we load
|
||||
// the file:// URI, otherwise we won't get the same web console.
|
||||
let remoteType = E10SUtils.getRemoteTypeForURI(uri.spec,
|
||||
gMultiProcessBrowser);
|
||||
let { browser } = yield loadTab("about:blank", remoteType);
|
||||
|
||||
hud = yield openConsole();
|
||||
hud.jsterm.clearOutput();
|
||||
|
|
|
@ -78,8 +78,8 @@ public:
|
|||
if (mNormalIter && mTrackingIter &&
|
||||
mNormalIter != mNormalStopAt &&
|
||||
mTrackingIter != mTrackingStopAt &&
|
||||
(mTrackingIter->mWhen < mNormalIter->mWhen ||
|
||||
(mTrackingIter->mWhen == mNormalIter->mWhen &&
|
||||
(mTrackingIter->When() < mNormalIter->When() ||
|
||||
(mTrackingIter->When() == mNormalIter->When() &&
|
||||
mTrackingIter->mTimeoutId < mNormalIter->mTimeoutId))) {
|
||||
timeout = mTrackingIter;
|
||||
mKind = Kind::Tracking;
|
||||
|
|
|
@ -108,5 +108,54 @@ Timeout::HasRefCnt(uint32_t aCount) const
|
|||
}
|
||||
#endif // DEBUG
|
||||
|
||||
void
|
||||
Timeout::SetWhenOrTimeRemaining(const TimeStamp& aBaseTime,
|
||||
const TimeDuration& aDelay)
|
||||
{
|
||||
// This must not be called on dummy timeouts. Instead use SetDummyWhen().
|
||||
MOZ_DIAGNOSTIC_ASSERT(mWindow);
|
||||
|
||||
// If we are frozen simply set mTimeRemaining to be the "time remaining" in
|
||||
// the timeout (i.e., the interval itself). This will be used to create a
|
||||
// new mWhen time when the window is thawed. The end effect is that time does
|
||||
// not appear to pass for frozen windows.
|
||||
if (mWindow->IsFrozen()) {
|
||||
mWhen = TimeStamp();
|
||||
mTimeRemaining = aDelay;
|
||||
return;
|
||||
}
|
||||
|
||||
// Since we are not frozen we must set a precise mWhen target wakeup
|
||||
// time. Even if we are suspended we want to use this target time so
|
||||
// that it appears time passes while suspended.
|
||||
mWhen = aBaseTime + aDelay;
|
||||
mTimeRemaining = TimeDuration(0);
|
||||
}
|
||||
|
||||
void
|
||||
Timeout::SetDummyWhen(const TimeStamp& aWhen)
|
||||
{
|
||||
MOZ_DIAGNOSTIC_ASSERT(!mWindow);
|
||||
mWhen = aWhen;
|
||||
}
|
||||
|
||||
const TimeStamp&
|
||||
Timeout::When() const
|
||||
{
|
||||
MOZ_DIAGNOSTIC_ASSERT(!mWhen.IsNull());
|
||||
// Note, mWindow->IsFrozen() can be true here. The Freeze() method calls
|
||||
// When() to calculate the delay to populate mTimeRemaining.
|
||||
return mWhen;
|
||||
}
|
||||
|
||||
const TimeDuration&
|
||||
Timeout::TimeRemaining() const
|
||||
{
|
||||
MOZ_DIAGNOSTIC_ASSERT(mWhen.IsNull());
|
||||
// Note, mWindow->IsFrozen() can be false here. The Thaw() method calls
|
||||
// TimeRemaining() to calculate the new When() value.
|
||||
return mTimeRemaining;
|
||||
}
|
||||
|
||||
} // namespace dom
|
||||
} // namespace mozilla
|
||||
|
|
|
@ -11,7 +11,6 @@
|
|||
#include "mozilla/TimeStamp.h"
|
||||
#include "nsCOMPtr.h"
|
||||
#include "nsCycleCollectionParticipant.h"
|
||||
#include "nsPIDOMWindow.h"
|
||||
|
||||
class nsGlobalWindow;
|
||||
class nsIEventTarget;
|
||||
|
@ -48,6 +47,17 @@ public:
|
|||
bool HasRefCnt(uint32_t aCount) const;
|
||||
#endif // DEBUG
|
||||
|
||||
void SetWhenOrTimeRemaining(const TimeStamp& aBaseTime,
|
||||
const TimeDuration& aDelay);
|
||||
|
||||
void SetDummyWhen(const TimeStamp& aWhen);
|
||||
|
||||
// Can only be called when not frozen.
|
||||
const TimeStamp& When() const;
|
||||
|
||||
// Can only be called when frozen.
|
||||
const TimeDuration& TimeRemaining() const;
|
||||
|
||||
// Window for which this timeout fires
|
||||
RefPtr<nsGlobalWindow> mWindow;
|
||||
|
||||
|
@ -71,14 +81,6 @@ public:
|
|||
// Interval in milliseconds
|
||||
uint32_t mInterval;
|
||||
|
||||
// mWhen and mTimeRemaining can't be in a union, sadly, because they
|
||||
// have constructors.
|
||||
// Nominal time to run this timeout. Use only when timeouts are not
|
||||
// suspended.
|
||||
TimeStamp mWhen;
|
||||
// Remaining time to wait. Used only when timeouts are suspended.
|
||||
TimeDuration mTimeRemaining;
|
||||
|
||||
// Principal with which to execute
|
||||
nsCOMPtr<nsIPrincipal> mPrincipal;
|
||||
|
||||
|
@ -95,6 +97,14 @@ public:
|
|||
nsCOMPtr<nsITimeoutHandler> mScriptHandler;
|
||||
|
||||
private:
|
||||
// mWhen and mTimeRemaining can't be in a union, sadly, because they
|
||||
// have constructors.
|
||||
// Nominal time to run this timeout. Use only when timeouts are not
|
||||
// frozen.
|
||||
TimeStamp mWhen;
|
||||
// Remaining time to wait. Used only when timeouts are frozen.
|
||||
TimeDuration mTimeRemaining;
|
||||
|
||||
~Timeout();
|
||||
};
|
||||
|
||||
|
|
|
@ -177,24 +177,14 @@ TimeoutManager::SetTimeout(nsITimeoutHandler* aHandler,
|
|||
realInterval = std::max(realInterval, uint32_t(DOMMinTimeoutValue()));
|
||||
}
|
||||
|
||||
TimeDuration delta = TimeDuration::FromMilliseconds(realInterval);
|
||||
timeout->mWindow = &mWindow;
|
||||
|
||||
if (mWindow.IsFrozen()) {
|
||||
// If we are frozen simply set timeout->mTimeRemaining to be the
|
||||
// "time remaining" in the timeout (i.e., the interval itself). This
|
||||
// will be used to create a new mWhen time when the window is thawed.
|
||||
// The end effect is that time does not appear to pass for frozen windows.
|
||||
timeout->mTimeRemaining = delta;
|
||||
} else {
|
||||
// Since we are not frozen we must set a precise mWhen target wakeup
|
||||
// time. Even if we are suspended we want to use this target time so
|
||||
// that it appears time passes while suspended.
|
||||
timeout->mWhen = TimeStamp::Now() + delta;
|
||||
}
|
||||
TimeDuration delta = TimeDuration::FromMilliseconds(realInterval);
|
||||
timeout->SetWhenOrTimeRemaining(TimeStamp::Now(), delta);
|
||||
|
||||
// If we're not suspended, then set the timer.
|
||||
if (!mWindow.IsSuspended()) {
|
||||
MOZ_ASSERT(!timeout->mWhen.IsNull());
|
||||
MOZ_ASSERT(!timeout->When().IsNull());
|
||||
|
||||
nsresult rv;
|
||||
timeout->mTimer = do_CreateInstance("@mozilla.org/timer;1", &rv);
|
||||
|
@ -214,8 +204,6 @@ TimeoutManager::SetTimeout(nsITimeoutHandler* aHandler,
|
|||
Unused << copy.forget();
|
||||
}
|
||||
|
||||
timeout->mWindow = &mWindow;
|
||||
|
||||
if (!aIsInterval) {
|
||||
timeout->mNestingLevel = nestingLevel;
|
||||
}
|
||||
|
@ -335,23 +323,23 @@ TimeoutManager::RunTimeout(Timeout* aTimeout)
|
|||
TimeStamp now = TimeStamp::Now();
|
||||
TimeStamp deadline;
|
||||
|
||||
if (aTimeout && aTimeout->mWhen > now) {
|
||||
if (aTimeout && aTimeout->When() > now) {
|
||||
// The OS timer fired early (which can happen due to the timers
|
||||
// having lower precision than TimeStamp does). Set |deadline| to
|
||||
// be the time when the OS timer *should* have fired so that any
|
||||
// timers that *should* have fired before aTimeout *will* be fired
|
||||
// now.
|
||||
|
||||
deadline = aTimeout->mWhen;
|
||||
deadline = aTimeout->When();
|
||||
} else {
|
||||
deadline = now;
|
||||
}
|
||||
|
||||
// The timeout list is kept in deadline order. Discover the latest timeout
|
||||
// whose deadline has expired. On some platforms, native timeout events fire
|
||||
// "early", but we handled that above by setting deadline to aTimeout->mWhen
|
||||
// "early", but we handled that above by setting deadline to aTimeout->When()
|
||||
// if the timer fired early. So we can stop walking if we get to timeouts
|
||||
// whose mWhen is greater than deadline, since once that happens we know
|
||||
// whose When() is greater than deadline, since once that happens we know
|
||||
// nothing past that point is expired.
|
||||
{
|
||||
// Use a nested scope in order to make sure the strong references held by
|
||||
|
@ -362,7 +350,7 @@ TimeoutManager::RunTimeout(Timeout* aTimeout)
|
|||
nullptr);
|
||||
while (true) {
|
||||
Timeout* timeout = expiredIter.Next();
|
||||
if (!timeout || timeout->mWhen > deadline) {
|
||||
if (!timeout || timeout->When() > deadline) {
|
||||
break;
|
||||
}
|
||||
|
||||
|
@ -410,14 +398,14 @@ TimeoutManager::RunTimeout(Timeout* aTimeout)
|
|||
// list for any timeouts inserted as a result of running a timeout.
|
||||
RefPtr<Timeout> dummy_normal_timeout = new Timeout();
|
||||
dummy_normal_timeout->mFiringDepth = firingDepth;
|
||||
dummy_normal_timeout->mWhen = now;
|
||||
dummy_normal_timeout->SetDummyWhen(now);
|
||||
if (last_expired_timeout_is_normal) {
|
||||
last_expired_normal_timeout->setNext(dummy_normal_timeout);
|
||||
}
|
||||
|
||||
RefPtr<Timeout> dummy_tracking_timeout = new Timeout();
|
||||
dummy_tracking_timeout->mFiringDepth = firingDepth;
|
||||
dummy_tracking_timeout->mWhen = now;
|
||||
dummy_tracking_timeout->SetDummyWhen(now);
|
||||
if (!last_expired_timeout_is_normal) {
|
||||
last_expired_tracking_timeout->setNext(dummy_tracking_timeout);
|
||||
}
|
||||
|
@ -702,7 +690,7 @@ TimeoutManager::RescheduleTimeout(Timeout* aTimeout, const TimeStamp& now,
|
|||
if (aRunningPendingTimeouts) {
|
||||
firingTime = now + nextInterval;
|
||||
} else {
|
||||
firingTime = aTimeout->mWhen + nextInterval;
|
||||
firingTime = aTimeout->When() + nextInterval;
|
||||
}
|
||||
|
||||
TimeStamp currentNow = TimeStamp::Now();
|
||||
|
@ -715,26 +703,13 @@ TimeoutManager::RescheduleTimeout(Timeout* aTimeout, const TimeStamp& now,
|
|||
delay = TimeDuration(0);
|
||||
}
|
||||
|
||||
aTimeout->SetWhenOrTimeRemaining(currentNow, delay);
|
||||
|
||||
if (!aTimeout->mTimer) {
|
||||
if (mWindow.IsFrozen()) {
|
||||
// If we are frozen simply set timeout->mTimeRemaining to be the
|
||||
// "time remaining" in the timeout (i.e., the interval itself). This
|
||||
// will be used to create a new mWhen time when the window is thawed.
|
||||
// The end effect is that time does not appear to pass for frozen windows.
|
||||
aTimeout->mTimeRemaining = delay;
|
||||
} else if (mWindow.IsSuspended()) {
|
||||
// Since we are not frozen we must set a precise mWhen target wakeup
|
||||
// time. Even if we are suspended we want to use this target time so
|
||||
// that it appears time passes while suspended.
|
||||
aTimeout->mWhen = currentNow + delay;
|
||||
} else {
|
||||
MOZ_ASSERT_UNREACHABLE("Window should be frozen or suspended.");
|
||||
}
|
||||
MOZ_DIAGNOSTIC_ASSERT(mWindow.IsFrozen() || mWindow.IsSuspended());
|
||||
return true;
|
||||
}
|
||||
|
||||
aTimeout->mWhen = currentNow + delay;
|
||||
|
||||
// Reschedule the OS timer. Don't bother returning any error codes if
|
||||
// this fails since the callers of this method don't care about them.
|
||||
nsresult rv = aTimeout->InitTimer(mWindow.EventTargetFor(TaskCategory::Timer),
|
||||
|
@ -806,7 +781,7 @@ TimeoutManager::Timeouts::ResetTimersForThrottleReduction(int32_t aPreviousThrot
|
|||
|
||||
// If insertion point is non-null, we're in the middle of firing timers and
|
||||
// the timers we're planning to fire all come before insertion point;
|
||||
// insertion point itself is a dummy timeout with an mWhen that may be
|
||||
// insertion point itself is a dummy timeout with an When() that may be
|
||||
// semi-bogus. In that case, we don't need to do anything with insertion
|
||||
// point or anything before it, so should start at the timer after insertion
|
||||
// point, if there is one.
|
||||
|
@ -816,15 +791,15 @@ TimeoutManager::Timeouts::ResetTimersForThrottleReduction(int32_t aPreviousThrot
|
|||
timeout; ) {
|
||||
// It's important that this check be <= so that we guarantee that
|
||||
// taking std::max with |now| won't make a quantity equal to
|
||||
// timeout->mWhen below.
|
||||
if (timeout->mWhen <= now) {
|
||||
// timeout->When() below.
|
||||
if (timeout->When() <= now) {
|
||||
timeout = timeout->getNext();
|
||||
continue;
|
||||
}
|
||||
|
||||
if (timeout->mWhen - now >
|
||||
if (timeout->When() - now >
|
||||
TimeDuration::FromMilliseconds(aPreviousThrottleDelayMS)) {
|
||||
// No need to loop further. Timeouts are sorted in mWhen order
|
||||
// No need to loop further. Timeouts are sorted in When() order
|
||||
// and the ones after this point were all set up for at least
|
||||
// gMinBackgroundTimeoutValue ms and hence were not clamped.
|
||||
break;
|
||||
|
@ -842,26 +817,27 @@ TimeoutManager::Timeouts::ResetTimersForThrottleReduction(int32_t aPreviousThrot
|
|||
if (oldInterval > interval) {
|
||||
// unclamp
|
||||
TimeStamp firingTime =
|
||||
std::max(timeout->mWhen - oldInterval + interval, now);
|
||||
std::max(timeout->When() - oldInterval + interval, now);
|
||||
|
||||
NS_ASSERTION(firingTime < timeout->mWhen,
|
||||
NS_ASSERTION(firingTime < timeout->When(),
|
||||
"Our firing time should strictly decrease!");
|
||||
|
||||
TimeDuration delay = firingTime - now;
|
||||
timeout->mWhen = firingTime;
|
||||
timeout->SetWhenOrTimeRemaining(now, delay);
|
||||
MOZ_DIAGNOSTIC_ASSERT(timeout->When() == firingTime);
|
||||
|
||||
// Since we reset mWhen we need to move |timeout| to the right
|
||||
// place in the list so that it remains sorted by mWhen.
|
||||
// Since we reset When() we need to move |timeout| to the right
|
||||
// place in the list so that it remains sorted by When().
|
||||
|
||||
// Get the pointer to the next timeout now, before we move the
|
||||
// current timeout in the list.
|
||||
Timeout* nextTimeout = timeout->getNext();
|
||||
|
||||
// It is safe to remove and re-insert because mWhen is now
|
||||
// It is safe to remove and re-insert because When() is now
|
||||
// strictly smaller than it used to be, so we know we'll insert
|
||||
// |timeout| before nextTimeout.
|
||||
NS_ASSERTION(!nextTimeout ||
|
||||
timeout->mWhen < nextTimeout->mWhen, "How did that happen?");
|
||||
timeout->When() < nextTimeout->When(), "How did that happen?");
|
||||
timeout->remove();
|
||||
// Insert() will addref |timeout| and reset mFiringDepth. Make sure to
|
||||
// undo that after calling it.
|
||||
|
@ -937,10 +913,10 @@ TimeoutManager::Timeouts::Insert(Timeout* aTimeout, SortBy aSortBy)
|
|||
for (prevSibling = GetLast();
|
||||
prevSibling && prevSibling != InsertionPoint() &&
|
||||
// This condition needs to match the one in SetTimeoutOrInterval that
|
||||
// determines whether to set mWhen or mTimeRemaining.
|
||||
// determines whether to set When() or TimeRemaining().
|
||||
(aSortBy == SortBy::TimeRemaining ?
|
||||
prevSibling->mTimeRemaining > aTimeout->mTimeRemaining :
|
||||
prevSibling->mWhen > aTimeout->mWhen);
|
||||
prevSibling->TimeRemaining() > aTimeout->TimeRemaining() :
|
||||
prevSibling->When() > aTimeout->When());
|
||||
prevSibling = prevSibling->getPrevious()) {
|
||||
/* Do nothing; just searching */
|
||||
}
|
||||
|
@ -1028,15 +1004,15 @@ TimeoutManager::Resume()
|
|||
|
||||
MOZ_ASSERT(!aTimeout->mTimer);
|
||||
|
||||
// The timeout mWhen is set to the absolute time when the timer should
|
||||
// The timeout When() is set to the absolute time when the timer should
|
||||
// fire. Recalculate the delay from now until that deadline. If the
|
||||
// the deadline has already passed or falls within our minimum delay
|
||||
// deadline, then clamp the resulting value to the minimum delay. The
|
||||
// mWhen will remain at its absolute time, but we won'aTimeout fire the OS
|
||||
// When() will remain at its absolute time, but we won'aTimeout fire the OS
|
||||
// timer until our calculated delay has passed.
|
||||
int32_t remaining = 0;
|
||||
if (aTimeout->mWhen > now) {
|
||||
remaining = static_cast<int32_t>((aTimeout->mWhen - now).ToMilliseconds());
|
||||
if (aTimeout->When() > now) {
|
||||
remaining = static_cast<int32_t>((aTimeout->When() - now).ToMilliseconds());
|
||||
}
|
||||
uint32_t delay = std::max(remaining, DOMMinTimeoutValue());
|
||||
|
||||
|
@ -1068,11 +1044,12 @@ TimeoutManager::Freeze()
|
|||
// re-apply it when the window is Thaw()'d. This effectively
|
||||
// shifts timers to the right as if time does not pass while
|
||||
// the window is frozen.
|
||||
if (aTimeout->mWhen > now) {
|
||||
aTimeout->mTimeRemaining = aTimeout->mWhen - now;
|
||||
} else {
|
||||
aTimeout->mTimeRemaining = TimeDuration(0);
|
||||
TimeDuration delta(0);
|
||||
if (aTimeout->When() > now) {
|
||||
delta = aTimeout->When() - now;
|
||||
}
|
||||
aTimeout->SetWhenOrTimeRemaining(now, delta);
|
||||
MOZ_DIAGNOSTIC_ASSERT(aTimeout->TimeRemaining() == delta);
|
||||
|
||||
// Since we are suspended there should be no OS timer set for
|
||||
// this timeout entry.
|
||||
|
@ -1096,8 +1073,9 @@ TimeoutManager::Thaw()
|
|||
return;
|
||||
}
|
||||
|
||||
// Set mWhen back to the time when the timer is supposed to fire.
|
||||
aTimeout->mWhen = now + aTimeout->mTimeRemaining;
|
||||
// Set When() back to the time when the timer is supposed to fire.
|
||||
aTimeout->SetWhenOrTimeRemaining(now, aTimeout->TimeRemaining());
|
||||
MOZ_DIAGNOSTIC_ASSERT(!aTimeout->When().IsNull());
|
||||
|
||||
MOZ_ASSERT(!aTimeout->mTimer);
|
||||
});
|
||||
|
|
|
@ -874,9 +874,6 @@ nsOuterWindowProxy::getOwnPropertyDescriptor(JSContext* cx,
|
|||
}
|
||||
// else fall through to js::Wrapper
|
||||
|
||||
// When we change this to always claim the property is configurable (bug
|
||||
// 1178639), update the comments in nsOuterWindowProxy::defineProperty
|
||||
// accordingly.
|
||||
return js::Wrapper::getOwnPropertyDescriptor(cx, proxy, id, desc);
|
||||
}
|
||||
|
||||
|
@ -894,29 +891,6 @@ nsOuterWindowProxy::defineProperty(JSContext* cx,
|
|||
return result.failCantDefineWindowElement();
|
||||
}
|
||||
|
||||
#ifndef RELEASE_OR_BETA // To be turned on in bug 1178638.
|
||||
// For now, allow chrome code to define non-configurable properties
|
||||
// on windows, until we sort out what exactly the addon SDK is
|
||||
// doing. In the meantime, this still allows us to test web compat
|
||||
// behavior.
|
||||
if (desc.hasConfigurable() && !desc.configurable() &&
|
||||
!nsContentUtils::IsCallerChrome()) {
|
||||
return ThrowErrorMessage(cx, MSG_DEFINE_NON_CONFIGURABLE_PROP_ON_WINDOW);
|
||||
}
|
||||
|
||||
// Note that if hasConfigurable() is false we do NOT want to
|
||||
// setConfigurable(true). That would make this code:
|
||||
//
|
||||
// var x;
|
||||
// window.x = 5;
|
||||
//
|
||||
// fail, because the JS engine ends up converting the assignment into a define
|
||||
// with !hasConfigurable(), but the var actually declared a non-configurable
|
||||
// property on our underlying Window object, so the set would fail if we
|
||||
// forced setConfigurable(true) here. What we want to do instead is change
|
||||
// getOwnPropertyDescriptor to always claim configurable. See bug 1178639.
|
||||
#endif
|
||||
|
||||
return js::Wrapper::defineProperty(cx, proxy, id, desc, result);
|
||||
}
|
||||
|
||||
|
|
|
@ -779,7 +779,6 @@ skip-if = toolkit == 'android'
|
|||
[test_window_constructor.html]
|
||||
[test_window_cross_origin_props.html]
|
||||
[test_window_define_nonconfigurable.html]
|
||||
skip-if = release_or_beta
|
||||
[test_window_define_symbol.html]
|
||||
[test_window_element_enumeration.html]
|
||||
[test_window_enumeration.html]
|
||||
|
|
|
@ -10,31 +10,78 @@ https://bugzilla.mozilla.org/show_bug.cgi?id=1107443
|
|||
<link rel="stylesheet" type="text/css" href="/tests/SimpleTest/test.css"/>
|
||||
<script type="application/javascript">
|
||||
|
||||
/** Test for Bug 1107443 **/
|
||||
try {
|
||||
Object.defineProperty(window, "nosuchprop", { value: 5, configurable: false });
|
||||
throw "didn't throw";
|
||||
} catch (e) {
|
||||
is(e instanceof TypeError, true,
|
||||
"defineProperty(window) with a non-configurable property should " +
|
||||
"throw a TypeError, instead got: " + e);
|
||||
is(Object.getOwnPropertyDescriptor(window, "nosuchprop"), undefined,
|
||||
'Window should not have property after an attempt to define it failed');
|
||||
}
|
||||
|
||||
Object.defineProperty(window, "nosuchprop", { value: 6 });
|
||||
/**
|
||||
* Test for Bug 1107443, modified when it was backed out in bug 1329323.
|
||||
* This is now testing the _current_ behavior, not the desired one; expect
|
||||
* failures in this test and needing to update it when bug 1329324 is
|
||||
* fixed.
|
||||
*/
|
||||
var retval = Object.defineProperty(window, "nosuchprop",
|
||||
{ value: 5, configurable: false });
|
||||
todo_is(retval, false,
|
||||
"Should return false when 'failing' to define non-configurable property via Object.defineProperty.")
|
||||
var desc = Object.getOwnPropertyDescriptor(window, "nosuchprop");
|
||||
is(typeof(desc), "object", "Should have a property now");
|
||||
todo_is(desc.configurable, true, "Property should be configurable");
|
||||
is(desc.writable, false, "Property should be readonly");
|
||||
is(desc.value, 6, "Property should have the right value");
|
||||
is(typeof(desc), "object", "Should have a property 'nosuchprop' now");
|
||||
todo_is(desc.configurable, true,
|
||||
"Property 'nosuchprop' should be configurable");
|
||||
is(desc.writable, false, "Property 'nosuchprop' should be readonly");
|
||||
is(desc.value, 5, "Property 'nosuchprop' should have the right value");
|
||||
|
||||
Object.defineProperty(window, "nosuchprop2", { value: 7, configurable: true });
|
||||
retval = Object.defineProperty(window, "nosuchprop2", { value: 6 });
|
||||
is(retval, window,
|
||||
"Should return object when succesfully defining 'nosuchprop2'");
|
||||
desc = Object.getOwnPropertyDescriptor(window, "nosuchprop2");
|
||||
is(typeof(desc), "object", "Should have a property now");
|
||||
is(desc.configurable, true, "Property should be configurable");
|
||||
is(desc.writable, false, "Property should be readonly");
|
||||
is(desc.value, 7, "Property should have the right value");
|
||||
is(typeof(desc), "object", "Should have a property 'nosuchprop2' now");
|
||||
todo_is(desc.configurable, true,
|
||||
"Property 'nosuchprop2' should be configurable");
|
||||
is(desc.writable, false, "Property 'nosuchprop2' should be readonly");
|
||||
is(desc.value, 6, "Property 'nosuchprop2' should have the right value");
|
||||
|
||||
retval = Object.defineProperty(window, "nosuchprop3",
|
||||
{ value: 7, configurable: true });
|
||||
is(retval, window,
|
||||
"Should return object when succesfully defining 'nosuchprop3'");
|
||||
desc = Object.getOwnPropertyDescriptor(window, "nosuchprop3");
|
||||
is(typeof(desc), "object", "Should have a property 'nosuchprop3' now");
|
||||
is(desc.configurable, true,
|
||||
"Property 'nosuchprop3' should be configurable");
|
||||
is(desc.writable, false, "Property 'nosuchprop3' should be readonly");
|
||||
is(desc.value, 7, "Property 'nosuchprop3' should have the right value");
|
||||
|
||||
// XXXbz it's not actually entirely clear what behavior the
|
||||
// Reflect.defineProperty bits should have. Check it carefully once there's a
|
||||
// spec.
|
||||
retval = Reflect.defineProperty(window, "nosuchprop4",
|
||||
{ value: 8, configurable: false });
|
||||
todo_is(retval, false,
|
||||
"Should not be able to Reflect.defineProperty if non-configurable");
|
||||
desc = Object.getOwnPropertyDescriptor(window, "nosuchprop4");
|
||||
is(typeof(desc), "object", "Should have a property 'nosuchprop4' now");
|
||||
todo_is(desc.configurable, true,
|
||||
"Property 'nosuchprop4' should be configurable");
|
||||
is(desc.writable, false, "Property 'nosuchprop4' should be readonly");
|
||||
is(desc.value, 8, "Property 'nosuchprop4' should have the right value");
|
||||
|
||||
retval = Reflect.defineProperty(window, "nosuchprop5",
|
||||
{ value: 9 });
|
||||
is(retval, true,
|
||||
"Should be able to Reflect.defineProperty with default configurability");
|
||||
desc = Object.getOwnPropertyDescriptor(window, "nosuchprop5");
|
||||
is(typeof(desc), "object", "Should have a property 'nosuchprop5' now");
|
||||
todo_is(desc.configurable, true,
|
||||
"Property 'nosuchprop5' should be configurable");
|
||||
is(desc.writable, false, "Property 'nosuchprop5' should be readonly");
|
||||
is(desc.value, 9, "Property 'nosuchprop5' should have the right value");
|
||||
|
||||
retval = Reflect.defineProperty(window, "nosuchprop6",
|
||||
{ value: 10, configurable: true });
|
||||
is(retval, true,
|
||||
"Should be able to Reflect.defineProperty if configurable");
|
||||
desc = Object.getOwnPropertyDescriptor(window, "nosuchprop6");
|
||||
is(typeof(desc), "object", "Should have a property 'nosuchprop6' now");
|
||||
is(desc.configurable, true, "Property 'nosuchprop6' should be configurable");
|
||||
is(desc.writable, false, "Property 'nosuchprop6' should be readonly");
|
||||
is(desc.value, 10, "Property 'nosuchprop6' should have the right value");
|
||||
</script>
|
||||
</head>
|
||||
<body>
|
||||
|
|
|
@ -1375,10 +1375,16 @@ def UnionTypes(unionTypes, config):
|
|||
# And if it needs rooting, we need RootedDictionary too
|
||||
if typeNeedsRooting(f):
|
||||
headers.add("mozilla/dom/RootedDictionary.h")
|
||||
elif f.isFloat() and not f.isUnrestricted():
|
||||
# Restricted floats are tested for finiteness
|
||||
implheaders.add("mozilla/FloatingPoint.h")
|
||||
implheaders.add("mozilla/dom/PrimitiveConversions.h")
|
||||
elif f.isEnum():
|
||||
# Need to see the actual definition of the enum,
|
||||
# unfortunately.
|
||||
headers.add(CGHeaders.getDeclarationFilename(f.inner))
|
||||
elif f.isPrimitive():
|
||||
implheaders.add("mozilla/dom/PrimitiveConversions.h")
|
||||
elif f.isCallback():
|
||||
# Callbacks always use strong refs, so we need to include
|
||||
# the right header to be able to Release() in our inlined
|
||||
|
@ -1446,6 +1452,10 @@ def UnionConversions(unionTypes, config):
|
|||
headers.add(CGHeaders.getDeclarationFilename(f.inner))
|
||||
elif f.isDictionary():
|
||||
headers.add(CGHeaders.getDeclarationFilename(f.inner))
|
||||
elif f.isFloat() and not f.isUnrestricted():
|
||||
# Restricted floats are tested for finiteness
|
||||
headers.add("mozilla/FloatingPoint.h")
|
||||
headers.add("mozilla/dom/PrimitiveConversions.h")
|
||||
elif f.isPrimitive():
|
||||
headers.add("mozilla/dom/PrimitiveConversions.h")
|
||||
elif f.isMozMap():
|
||||
|
@ -16786,7 +16796,6 @@ class GlobalGenRoots():
|
|||
# If it stops being inlined or stops calling CallerSubsumes
|
||||
# both this bit and the bit in CGBindingRoot can be removed.
|
||||
includes.add("mozilla/dom/BindingUtils.h")
|
||||
implincludes.add("mozilla/dom/PrimitiveConversions.h")
|
||||
|
||||
# Wrap all of that in our namespaces.
|
||||
curr = CGNamespace.build(['mozilla', 'dom'], unions)
|
||||
|
|
|
@ -100,3 +100,4 @@ MSG_DEF(MSG_TIME_VALUE_OUT_OF_RANGE, 1, JSEXN_TYPEERR, "{0} is outside the suppo
|
|||
MSG_DEF(MSG_ONLY_IF_CACHED_WITHOUT_SAME_ORIGIN, 1, JSEXN_TYPEERR, "Request mode '{0}' was used, but request cache mode 'only-if-cached' can only be used with request mode 'same-origin'.")
|
||||
MSG_DEF(MSG_THRESHOLD_RANGE_ERROR, 0, JSEXN_RANGEERR, "Threshold values must all be in the range [0, 1].")
|
||||
MSG_DEF(MSG_WORKER_THREAD_SHUTTING_DOWN, 0, JSEXN_TYPEERR, "The Worker thread is shutting down.")
|
||||
MSG_DEF(MSG_CACHE_OPEN_FAILED, 0, JSEXN_TYPEERR, "CacheStorage.open() failed to access the storage system.")
|
||||
|
|
|
@ -51,21 +51,12 @@ struct GCPolicy<mozilla::OwningNonNull<T>>
|
|||
} // namespace JS
|
||||
|
||||
namespace js {
|
||||
template<typename T>
|
||||
struct RootedBase<mozilla::OwningNonNull<T>>
|
||||
template<typename T, typename Wrapper>
|
||||
struct WrappedPtrOperations<mozilla::OwningNonNull<T>, Wrapper>
|
||||
{
|
||||
typedef mozilla::OwningNonNull<T> SmartPtrType;
|
||||
|
||||
operator SmartPtrType& () const
|
||||
{
|
||||
auto& self = *static_cast<const JS::Rooted<SmartPtrType>*>(this);
|
||||
return self.get();
|
||||
}
|
||||
|
||||
operator T& () const
|
||||
{
|
||||
auto& self = *static_cast<const JS::Rooted<SmartPtrType>*>(this);
|
||||
return self.get();
|
||||
return static_cast<const Wrapper*>(this)->get();
|
||||
}
|
||||
};
|
||||
} // namespace js
|
||||
|
|
|
@ -39,19 +39,12 @@ struct GCPolicy<RefPtr<T>>
|
|||
} // namespace JS
|
||||
|
||||
namespace js {
|
||||
template<typename T>
|
||||
struct RootedBase<RefPtr<T>>
|
||||
template<typename T, typename Wrapper>
|
||||
struct WrappedPtrOperations<RefPtr<T>, Wrapper>
|
||||
{
|
||||
operator RefPtr<T>& () const
|
||||
{
|
||||
auto& self = *static_cast<const JS::Rooted<RefPtr<T>>*>(this);
|
||||
return self.get();
|
||||
}
|
||||
|
||||
operator T*() const
|
||||
{
|
||||
auto& self = *static_cast<const JS::Rooted<RefPtr<T>>*>(this);
|
||||
return self.get();
|
||||
return static_cast<const Wrapper*>(this)->get();
|
||||
}
|
||||
};
|
||||
} // namespace js
|
||||
|
|
|
@ -155,6 +155,17 @@ CacheOpChild::Recv__delete__(const ErrorResult& aRv,
|
|||
{
|
||||
auto actor = static_cast<CacheChild*>(
|
||||
aResult.get_StorageOpenResult().actorChild());
|
||||
|
||||
// If we have a success status then we should have an actor. Gracefully
|
||||
// reject instead of crashing, though, if we get a nullptr here.
|
||||
MOZ_DIAGNOSTIC_ASSERT(actor);
|
||||
if (!actor) {
|
||||
ErrorResult status;
|
||||
status.ThrowTypeError<MSG_CACHE_OPEN_FAILED>();
|
||||
mPromise->MaybeReject(status);
|
||||
break;
|
||||
}
|
||||
|
||||
actor->SetWorkerHolder(GetWorkerHolder());
|
||||
RefPtr<Cache> cache = new Cache(mGlobal, actor);
|
||||
mPromise->MaybeResolve(cache);
|
||||
|
|
|
@ -501,6 +501,8 @@ void
|
|||
CacheStorage::ActorCreated(PBackgroundChild* aActor)
|
||||
{
|
||||
NS_ASSERT_OWNINGTHREAD(CacheStorage);
|
||||
MOZ_DIAGNOSTIC_ASSERT(NS_SUCCEEDED(mStatus));
|
||||
MOZ_DIAGNOSTIC_ASSERT(!mActor);
|
||||
MOZ_DIAGNOSTIC_ASSERT(aActor);
|
||||
|
||||
if (NS_WARN_IF(mWorkerHolder && mWorkerHolder->Notified())) {
|
||||
|
@ -511,19 +513,22 @@ CacheStorage::ActorCreated(PBackgroundChild* aActor)
|
|||
// WorkerHolder ownership is passed to the CacheStorageChild actor and any
|
||||
// actors it may create. The WorkerHolder will keep the worker thread alive
|
||||
// until the actors can gracefully shutdown.
|
||||
CacheStorageChild* newActor = new CacheStorageChild(this, mWorkerHolder);
|
||||
PCacheStorageChild* constructedActor =
|
||||
aActor->SendPCacheStorageConstructor(newActor, mNamespace, *mPrincipalInfo);
|
||||
mActor = new CacheStorageChild(this, mWorkerHolder);
|
||||
mWorkerHolder = nullptr;
|
||||
|
||||
if (NS_WARN_IF(!constructedActor)) {
|
||||
ActorFailed();
|
||||
// Pass the actor construction message to the parent. Note, if this fails
|
||||
// we can get DestroyInternal() and ActorFailed() called synchronously. This
|
||||
// will null out mActor and set an error mStatus.
|
||||
PCacheStorageChild* constructedActor =
|
||||
aActor->SendPCacheStorageConstructor(mActor, mNamespace, *mPrincipalInfo);
|
||||
|
||||
if (NS_WARN_IF(NS_FAILED(mStatus))) {
|
||||
MOZ_DIAGNOSTIC_ASSERT(!mActor);
|
||||
return;
|
||||
}
|
||||
|
||||
mWorkerHolder = nullptr;
|
||||
|
||||
MOZ_DIAGNOSTIC_ASSERT(constructedActor == newActor);
|
||||
mActor = newActor;
|
||||
MOZ_DIAGNOSTIC_ASSERT(mActor);
|
||||
MOZ_DIAGNOSTIC_ASSERT(constructedActor == mActor);
|
||||
|
||||
MaybeRunPendingRequests();
|
||||
MOZ_DIAGNOSTIC_ASSERT(mPendingRequests.IsEmpty());
|
||||
|
@ -533,7 +538,7 @@ void
|
|||
CacheStorage::ActorFailed()
|
||||
{
|
||||
NS_ASSERT_OWNINGTHREAD(CacheStorage);
|
||||
MOZ_DIAGNOSTIC_ASSERT(!NS_FAILED(mStatus));
|
||||
MOZ_DIAGNOSTIC_ASSERT(NS_SUCCEEDED(mStatus));
|
||||
|
||||
mStatus = NS_ERROR_UNEXPECTED;
|
||||
mWorkerHolder = nullptr;
|
||||
|
|
|
@ -85,6 +85,11 @@ CacheStreamControlParent::ActorDestroy(ActorDestroyReason aReason)
|
|||
{
|
||||
NS_ASSERT_OWNINGTHREAD(CacheStreamControlParent);
|
||||
CloseAllReadStreamsWithoutReporting();
|
||||
// If the initial SendPStreamControlConstructor() fails we will
|
||||
// be called before mStreamList is set.
|
||||
if (!mStreamList) {
|
||||
return;
|
||||
}
|
||||
mStreamList->RemoveStreamControl(this);
|
||||
mStreamList->NoteClosedAll();
|
||||
mStreamList = nullptr;
|
||||
|
|
|
@ -1215,6 +1215,7 @@ public:
|
|||
&cacheFound, &mCacheId);
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) { return rv; }
|
||||
if (cacheFound) {
|
||||
MOZ_DIAGNOSTIC_ASSERT(mCacheId != INVALID_CACHE_ID);
|
||||
return rv;
|
||||
}
|
||||
|
||||
|
@ -1227,12 +1228,14 @@ public:
|
|||
rv = trans.Commit();
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) { return rv; }
|
||||
|
||||
MOZ_DIAGNOSTIC_ASSERT(mCacheId != INVALID_CACHE_ID);
|
||||
return rv;
|
||||
}
|
||||
|
||||
virtual void
|
||||
Complete(Listener* aListener, ErrorResult&& aRv) override
|
||||
{
|
||||
MOZ_DIAGNOSTIC_ASSERT(aRv.Failed() || mCacheId != INVALID_CACHE_ID);
|
||||
aListener->OnOpComplete(Move(aRv), StorageOpenResult(), mCacheId);
|
||||
}
|
||||
|
||||
|
|
|
@ -1292,6 +1292,7 @@ CrossProcessSafeEvent(const WidgetEvent& aEvent)
|
|||
case eContextMenu:
|
||||
case eMouseEnterIntoWidget:
|
||||
case eMouseExitFromWidget:
|
||||
case eMouseTouchDrag:
|
||||
return true;
|
||||
default:
|
||||
return false;
|
||||
|
|
|
@ -193,6 +193,39 @@ uint32_t
|
|||
GetMajorStorageVersion(int32_t aStorageVersion)
|
||||
{
|
||||
return uint32_t(aStorageVersion >> 16);
|
||||
|
||||
}
|
||||
|
||||
nsresult
|
||||
CreateTables(mozIStorageConnection* aConnection)
|
||||
{
|
||||
AssertIsOnIOThread();
|
||||
MOZ_ASSERT(aConnection);
|
||||
|
||||
// The database doesn't have any tables for now. It's only used for storage
|
||||
// version checking.
|
||||
// However, this is the place where any future tables should be created.
|
||||
|
||||
nsresult rv;
|
||||
|
||||
#ifdef DEBUG
|
||||
{
|
||||
int32_t storageVersion;
|
||||
rv = aConnection->GetSchemaVersion(&storageVersion);
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
MOZ_ASSERT(storageVersion == 0);
|
||||
}
|
||||
#endif
|
||||
|
||||
rv = aConnection->SetSchemaVersion(kStorageVersion);
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
return NS_OK;
|
||||
}
|
||||
|
||||
/******************************************************************************
|
||||
|
@ -4094,12 +4127,25 @@ QuotaManager::MaybeRemoveOldDirectories()
|
|||
}
|
||||
|
||||
nsresult
|
||||
QuotaManager::UpgradeStorageFrom0ToCurrent(mozIStorageConnection* aConnection)
|
||||
QuotaManager::UpgradeStorageFrom0_0To1_0(mozIStorageConnection* aConnection)
|
||||
{
|
||||
AssertIsOnIOThread();
|
||||
MOZ_ASSERT(aConnection);
|
||||
|
||||
nsresult rv;
|
||||
nsresult rv = MaybeUpgradeIndexedDBDirectory();
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
rv = MaybeUpgradePersistentStorageDirectory();
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
rv = MaybeRemoveOldDirectories();
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
for (const PersistenceType persistenceType : kAllPersistenceTypes) {
|
||||
nsCOMPtr<nsIFile> directory =
|
||||
|
@ -4145,12 +4191,26 @@ QuotaManager::UpgradeStorageFrom0ToCurrent(mozIStorageConnection* aConnection)
|
|||
|
||||
#if 0
|
||||
nsresult
|
||||
QuotaManager::UpgradeStorageFrom1To2(mozIStorageConnection* aConnection)
|
||||
QuotaManager::UpgradeStorageFrom1_0To2_0(mozIStorageConnection* aConnection)
|
||||
{
|
||||
AssertIsOnIOThread();
|
||||
MOZ_ASSERT(aConnection);
|
||||
|
||||
nsresult rv = aConnection->SetSchemaVersion(MakeStorageVersion(2, 0));
|
||||
nsresult rv;
|
||||
|
||||
#ifdef DEBUG
|
||||
{
|
||||
int32_t storageVersion;
|
||||
rv = aConnection->GetSchemaVersion(&storageVersion);
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
MOZ_ASSERT(storageVersion == MakeStorageVersion(1, 0));
|
||||
}
|
||||
#endif
|
||||
|
||||
rv = aConnection->SetSchemaVersion(MakeStorageVersion(2, 0));
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
@ -4231,6 +4291,43 @@ QuotaManager::EnsureStorageIsInitialized()
|
|||
if (storageVersion < kStorageVersion) {
|
||||
const bool newDatabase = !storageVersion;
|
||||
|
||||
nsCOMPtr<nsIFile> storageDir =
|
||||
do_CreateInstance(NS_LOCAL_FILE_CONTRACTID, &rv);
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
rv = storageDir->InitWithPath(mStoragePath);
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
bool exists;
|
||||
rv = storageDir->Exists(&exists);
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
if (!exists) {
|
||||
nsCOMPtr<nsIFile> indexedDBDir =
|
||||
do_CreateInstance(NS_LOCAL_FILE_CONTRACTID, &rv);
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
rv = indexedDBDir->InitWithPath(mIndexedDBPath);
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
rv = indexedDBDir->Exists(&exists);
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
}
|
||||
|
||||
const bool newDirectory = !exists;
|
||||
|
||||
if (newDatabase) {
|
||||
// Set the page size first.
|
||||
if (kSQLitePageSizeOverride) {
|
||||
|
@ -4245,23 +4342,12 @@ QuotaManager::EnsureStorageIsInitialized()
|
|||
|
||||
mozStorageTransaction transaction(connection, false,
|
||||
mozIStorageConnection::TRANSACTION_IMMEDIATE);
|
||||
if (newDatabase) {
|
||||
rv = MaybeUpgradeIndexedDBDirectory();
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
rv = MaybeUpgradePersistentStorageDirectory();
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
rv = MaybeRemoveOldDirectories();
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
||||
rv = UpgradeStorageFrom0ToCurrent(connection);
|
||||
// An upgrade method can upgrade the database, the storage or both.
|
||||
// The upgrade loop below can only be avoided when there's no database and
|
||||
// no storage yet (e.g. new profile).
|
||||
if (newDatabase && newDirectory) {
|
||||
rv = CreateTables(connection);
|
||||
if (NS_WARN_IF(NS_FAILED(rv))) {
|
||||
return rv;
|
||||
}
|
||||
|
@ -4274,9 +4360,13 @@ QuotaManager::EnsureStorageIsInitialized()
|
|||
"Upgrade function needed due to storage version increase.");
|
||||
|
||||
while (storageVersion != kStorageVersion) {
|
||||
/* if (storageVersion == MakeStorageVersion(1, 0)) {
|
||||
rv = UpgradeStorageFrom1To2(connection);
|
||||
} else */ {
|
||||
if (storageVersion == 0) {
|
||||
rv = UpgradeStorageFrom0_0To1_0(connection);
|
||||
#if 0
|
||||
} else if (storageVersion == MakeStorageVersion(1, 0)) {
|
||||
rv = UpgradeStorageFrom1_0To2_0(connection);
|
||||
#endif
|
||||
} else {
|
||||
NS_WARNING("Unable to initialize storage, no upgrade path is "
|
||||
"available!");
|
||||
return NS_ERROR_FAILURE;
|
||||
|
|
|
@ -455,11 +455,11 @@ private:
|
|||
MaybeRemoveOldDirectories();
|
||||
|
||||
nsresult
|
||||
UpgradeStorageFrom0ToCurrent(mozIStorageConnection* aConnection);
|
||||
UpgradeStorageFrom0_0To1_0(mozIStorageConnection* aConnection);
|
||||
|
||||
#if 0
|
||||
nsresult
|
||||
UpgradeStorageFrom1To2(mozIStorageConnection* aConnection);
|
||||
UpgradeStorageFrom1_0To2_0(mozIStorageConnection* aConnection);
|
||||
#endif
|
||||
|
||||
nsresult
|
||||
|
|
|
@ -1579,13 +1579,17 @@ CacheCreator::DeleteCache()
|
|||
{
|
||||
AssertIsOnMainThread();
|
||||
|
||||
ErrorResult rv;
|
||||
// This is called when the load is canceled which can occur before
|
||||
// mCacheStorage is initialized.
|
||||
if (!mCacheStorage) {
|
||||
return;
|
||||
}
|
||||
|
||||
// It's safe to do this while Cache::Match() and Cache::Put() calls are
|
||||
// running.
|
||||
IgnoredErrorResult rv;
|
||||
RefPtr<Promise> promise = mCacheStorage->Delete(mCacheName, rv);
|
||||
if (NS_WARN_IF(rv.Failed())) {
|
||||
rv.SuppressException();
|
||||
return;
|
||||
}
|
||||
|
||||
|
|
|
@ -127,15 +127,15 @@ struct IsHeapConstructibleType<nsXBLMaybeCompiled<T>>
|
|||
static constexpr bool value = true;
|
||||
};
|
||||
|
||||
template <class UncompiledT>
|
||||
class HeapBase<nsXBLMaybeCompiled<UncompiledT>>
|
||||
template <class UncompiledT, class Wrapper>
|
||||
class HeapBase<nsXBLMaybeCompiled<UncompiledT>, Wrapper>
|
||||
{
|
||||
const JS::Heap<nsXBLMaybeCompiled<UncompiledT>>& wrapper() const {
|
||||
return *static_cast<const JS::Heap<nsXBLMaybeCompiled<UncompiledT>>*>(this);
|
||||
const Wrapper& wrapper() const {
|
||||
return *static_cast<const Wrapper*>(this);
|
||||
}
|
||||
|
||||
JS::Heap<nsXBLMaybeCompiled<UncompiledT>>& wrapper() {
|
||||
return *static_cast<JS::Heap<nsXBLMaybeCompiled<UncompiledT>>*>(this);
|
||||
Wrapper& wrapper() {
|
||||
return *static_cast<Wrapper*>(this);
|
||||
}
|
||||
|
||||
const nsXBLMaybeCompiled<UncompiledT>* extract() const {
|
||||
|
|
|
@ -1559,8 +1559,6 @@ public:
|
|||
bool aUseEmbeddedBitmap,
|
||||
bool aForceGDIMode);
|
||||
|
||||
static void UpdateSystemTextQuality();
|
||||
|
||||
private:
|
||||
static ID2D1Device *mD2D1Device;
|
||||
static ID3D11Device *mD3D11Device;
|
||||
|
|
|
@ -41,7 +41,6 @@
|
|||
#include "NativeFontResourceDWrite.h"
|
||||
#include <d3d10_1.h>
|
||||
#include "HelpersD2D.h"
|
||||
#include "HelpersWinFonts.h"
|
||||
#endif
|
||||
|
||||
#include "DrawTargetDual.h"
|
||||
|
@ -710,15 +709,6 @@ Factory::CreateDWriteGlyphRenderingOptions(IDWriteRenderingParams *aParams)
|
|||
return MakeAndAddRef<GlyphRenderingOptionsDWrite>(aParams);
|
||||
}
|
||||
|
||||
BYTE sSystemTextQuality = CLEARTYPE_QUALITY;
|
||||
void
|
||||
Factory::UpdateSystemTextQuality()
|
||||
{
|
||||
#ifdef WIN32
|
||||
gfx::UpdateSystemTextQuality();
|
||||
#endif
|
||||
}
|
||||
|
||||
uint64_t
|
||||
Factory::GetD2DVRAMUsageDrawTarget()
|
||||
{
|
||||
|
|
|
@ -6,45 +6,32 @@
|
|||
namespace mozilla {
|
||||
namespace gfx {
|
||||
|
||||
|
||||
extern BYTE sSystemTextQuality;
|
||||
|
||||
// Cleartype can be dynamically enabled/disabled, so we have to check it
|
||||
// everytime we want to render some text.
|
||||
static BYTE
|
||||
GetSystemTextQuality()
|
||||
{
|
||||
return sSystemTextQuality;
|
||||
}
|
||||
|
||||
// Cleartype can be dynamically enabled/disabled, so we have to allow for dynamically
|
||||
// updating it.
|
||||
static void
|
||||
UpdateSystemTextQuality()
|
||||
{
|
||||
BOOL font_smoothing;
|
||||
UINT smoothing_type;
|
||||
|
||||
if (!SystemParametersInfo(SPI_GETFONTSMOOTHING, 0, &font_smoothing, 0)) {
|
||||
sSystemTextQuality = DEFAULT_QUALITY;
|
||||
return;
|
||||
return DEFAULT_QUALITY;
|
||||
}
|
||||
|
||||
if (font_smoothing) {
|
||||
if (!SystemParametersInfo(SPI_GETFONTSMOOTHINGTYPE,
|
||||
0, &smoothing_type, 0)) {
|
||||
sSystemTextQuality = DEFAULT_QUALITY;
|
||||
return;
|
||||
return DEFAULT_QUALITY;
|
||||
}
|
||||
|
||||
if (smoothing_type == FE_FONTSMOOTHINGCLEARTYPE) {
|
||||
sSystemTextQuality = CLEARTYPE_QUALITY;
|
||||
return;
|
||||
return CLEARTYPE_QUALITY;
|
||||
}
|
||||
|
||||
sSystemTextQuality = ANTIALIASED_QUALITY;
|
||||
return;
|
||||
return ANTIALIASED_QUALITY;
|
||||
}
|
||||
|
||||
sSystemTextQuality = DEFAULT_QUALITY;
|
||||
return DEFAULT_QUALITY;
|
||||
}
|
||||
|
||||
static AntialiasMode
|
||||
|
|
|
@ -51,8 +51,38 @@ class TokenLexer : public Lexer
|
|||
|
||||
} // anonymous namespace
|
||||
|
||||
class MacroExpander::ScopedMacroReenabler final : angle::NonCopyable
|
||||
{
|
||||
public:
|
||||
ScopedMacroReenabler(MacroExpander *expander);
|
||||
~ScopedMacroReenabler();
|
||||
|
||||
private:
|
||||
MacroExpander *mExpander;
|
||||
};
|
||||
|
||||
MacroExpander::ScopedMacroReenabler::ScopedMacroReenabler(MacroExpander *expander)
|
||||
: mExpander(expander)
|
||||
{
|
||||
mExpander->mDeferReenablingMacros = true;
|
||||
}
|
||||
|
||||
MacroExpander::ScopedMacroReenabler::~ScopedMacroReenabler()
|
||||
{
|
||||
mExpander->mDeferReenablingMacros = false;
|
||||
for (auto *macro : mExpander->mMacrosToReenable)
|
||||
{
|
||||
macro->disabled = false;
|
||||
}
|
||||
mExpander->mMacrosToReenable.clear();
|
||||
}
|
||||
|
||||
MacroExpander::MacroExpander(Lexer *lexer, MacroSet *macroSet, Diagnostics *diagnostics)
|
||||
: mLexer(lexer), mMacroSet(macroSet), mDiagnostics(diagnostics), mTotalTokensInContexts(0)
|
||||
: mLexer(lexer),
|
||||
mMacroSet(macroSet),
|
||||
mDiagnostics(diagnostics),
|
||||
mTotalTokensInContexts(0),
|
||||
mDeferReenablingMacros(false)
|
||||
{
|
||||
}
|
||||
|
||||
|
@ -187,7 +217,14 @@ void MacroExpander::popMacro()
|
|||
ASSERT(context->empty());
|
||||
ASSERT(context->macro->disabled);
|
||||
ASSERT(context->macro->expansionCount > 0);
|
||||
if (mDeferReenablingMacros)
|
||||
{
|
||||
mMacrosToReenable.push_back(context->macro);
|
||||
}
|
||||
else
|
||||
{
|
||||
context->macro->disabled = false;
|
||||
}
|
||||
context->macro->expansionCount--;
|
||||
mTotalTokensInContexts -= context->replacements.size();
|
||||
delete context;
|
||||
|
@ -263,6 +300,11 @@ bool MacroExpander::collectMacroArgs(const Macro ¯o,
|
|||
|
||||
args->push_back(MacroArg());
|
||||
|
||||
// Defer reenabling macros until args collection is finished to avoid the possibility of
|
||||
// infinite recursion. Otherwise infinite recursion might happen when expanding the args after
|
||||
// macros have been popped from the context stack when parsing the args.
|
||||
ScopedMacroReenabler deferReenablingMacros(this);
|
||||
|
||||
int openParens = 1;
|
||||
while (openParens != 0)
|
||||
{
|
||||
|
|
|
@ -67,6 +67,11 @@ class MacroExpander : public Lexer
|
|||
std::unique_ptr<Token> mReserveToken;
|
||||
std::vector<MacroContext *> mContextStack;
|
||||
size_t mTotalTokensInContexts;
|
||||
|
||||
bool mDeferReenablingMacros;
|
||||
std::vector<const Macro *> mMacrosToReenable;
|
||||
|
||||
class ScopedMacroReenabler;
|
||||
};
|
||||
|
||||
} // namespace pp
|
||||
|
|
|
@ -971,3 +971,18 @@ TEST_F(DefineTest, UndefineInInvocationPreLParen)
|
|||
|
||||
preprocess(input, expected);
|
||||
}
|
||||
|
||||
// The name of the macro "a" is inside an incomplete macro invocation of macro "m()" in its own
|
||||
// expansion. This should not result in infinite recursion.
|
||||
TEST_F(DefineTest, RecursiveMacroNameInsideIncompleteMacroInvocationInMacroExpansion)
|
||||
{
|
||||
const char *input =
|
||||
"#define m(a)\n"
|
||||
"#define a m((a)\n"
|
||||
"a)\n";
|
||||
const char *expected =
|
||||
"\n"
|
||||
"\n"
|
||||
"\n";
|
||||
preprocess(input, expected);
|
||||
}
|
||||
|
|
|
@ -35,7 +35,6 @@
|
|||
#endif
|
||||
#ifdef XP_WIN
|
||||
#include "mozilla/gfx/DeviceManagerDx.h"
|
||||
#include "gfxDWriteFonts.h"
|
||||
#endif
|
||||
|
||||
namespace mozilla {
|
||||
|
@ -290,14 +289,6 @@ ClientLayerManager::EndTransactionInternal(DrawPaintedLayerCallback aCallback,
|
|||
{
|
||||
PaintTelemetry::AutoRecord record(PaintTelemetry::Metric::Rasterization);
|
||||
|
||||
#ifdef WIN32
|
||||
if (aCallbackData) {
|
||||
// Content processes don't get OnPaint called. So update here whenever we
|
||||
// may do Thebes drawing.
|
||||
gfxDWriteFont::UpdateClearTypeUsage();
|
||||
}
|
||||
#endif
|
||||
|
||||
PROFILER_LABEL("ClientLayerManager", "EndTransactionInternal",
|
||||
js::ProfileEntry::Category::GRAPHICS);
|
||||
|
||||
|
|
|
@ -0,0 +1,35 @@
|
|||
<!DOCTYPE html>
|
||||
<html>
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<style>
|
||||
#o_1 {
|
||||
position: absolute;
|
||||
top: 0px;
|
||||
width: 100px;
|
||||
height: 100px;
|
||||
background: red;
|
||||
}
|
||||
#o_2 {
|
||||
height: 10000px;
|
||||
}
|
||||
</style>
|
||||
<script>
|
||||
function boom(){
|
||||
let doc = document.documentElement;
|
||||
o_2.style.MozBorderEndStyle = "dotted";
|
||||
doc.style.MozPerspective = "24.5pt";
|
||||
o_0.style.MozTransformStyle = "preserve-3d";
|
||||
doc.style.overflow = "-moz-scrollbars-horizontal";
|
||||
doc.style.textOverflow = "''";
|
||||
o_0.style.offsetInlineStart = "calc(3*25px)";
|
||||
doc.style.paddingTop = "calc(67108864%)";
|
||||
doc.style.width = "3e-0%";
|
||||
o_0.style.display = "-moz-stack";
|
||||
o_0.style.position = "relative";
|
||||
}
|
||||
addEventListener("DOMContentLoaded", boom);
|
||||
</script>
|
||||
</head>
|
||||
<body id=o_0><div id=o_1></div><div id=o_2></div></body>
|
||||
</html>
|
|
@ -132,3 +132,4 @@ load 1216832-1.html
|
|||
load 1225125-1.html
|
||||
asserts-if(stylo,2) load 1308394.html # bug 1324669
|
||||
load 1317403-1.html
|
||||
load 1325159-1.html
|
||||
|
|
|
@ -123,7 +123,6 @@ gfxDWriteFont::~gfxDWriteFont()
|
|||
void
|
||||
gfxDWriteFont::UpdateClearTypeUsage()
|
||||
{
|
||||
Factory::UpdateSystemTextQuality();
|
||||
mUseClearType = UsingClearType();
|
||||
}
|
||||
|
||||
|
|
|
@ -538,13 +538,6 @@ bool Channel::ChannelImpl::ProcessIncomingMessages() {
|
|||
" with type " << m.type();
|
||||
#endif
|
||||
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
AutoSaveCurTraceInfo saveCurTraceInfo;
|
||||
SetCurTraceInfo(m.header()->source_event_id,
|
||||
m.header()->parent_task_id,
|
||||
m.header()->source_event_type);
|
||||
#endif
|
||||
|
||||
if (m.routing_id() == MSG_ROUTING_NONE &&
|
||||
m.type() == HELLO_MESSAGE_TYPE) {
|
||||
// The Hello message contains only the process id.
|
||||
|
|
|
@ -13,7 +13,7 @@
|
|||
#include "chrome/common/file_descriptor_set_posix.h"
|
||||
#endif
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
#include "GeckoTaskTracer.h"
|
||||
#include "GeckoTaskTracerImpl.h"
|
||||
#endif
|
||||
|
||||
#include "mozilla/Move.h"
|
||||
|
@ -145,4 +145,39 @@ uint32_t Message::num_fds() const {
|
|||
|
||||
#endif
|
||||
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
void *MessageTask() {
|
||||
return reinterpret_cast<void*>(&MessageTask);
|
||||
}
|
||||
|
||||
void
|
||||
Message::TaskTracerDispatch() {
|
||||
header()->task_id = GenNewUniqueTaskId();
|
||||
uintptr_t* vtab = reinterpret_cast<uintptr_t*>(&MessageTask);
|
||||
LogVirtualTablePtr(header()->task_id,
|
||||
header()->source_event_id,
|
||||
vtab);
|
||||
LogDispatch(header()->task_id,
|
||||
header()->parent_task_id,
|
||||
header()->source_event_id,
|
||||
header()->source_event_type);
|
||||
}
|
||||
|
||||
Message::AutoTaskTracerRun::AutoTaskTracerRun(Message& aMsg)
|
||||
: mMsg(aMsg)
|
||||
, mTaskId(mMsg.header()->task_id)
|
||||
, mSourceEventId(mMsg.header()->source_event_id) {
|
||||
LogBegin(mMsg.header()->task_id,
|
||||
mMsg.header()->source_event_id);
|
||||
SetCurTraceInfo(mMsg.header()->source_event_id,
|
||||
mMsg.header()->task_id,
|
||||
mMsg.header()->source_event_type);
|
||||
}
|
||||
|
||||
Message::AutoTaskTracerRun::~AutoTaskTracerRun() {
|
||||
AddLabel("IPC Message %s", mMsg.name());
|
||||
LogEnd(mTaskId, mSourceEventId);
|
||||
}
|
||||
#endif
|
||||
|
||||
} // namespace IPC
|
||||
|
|
|
@ -272,6 +272,19 @@ class Message : public Pickle {
|
|||
header()->flags |= INTERRUPT_BIT;
|
||||
}
|
||||
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
void TaskTracerDispatch();
|
||||
class AutoTaskTracerRun
|
||||
: public mozilla::tasktracer::AutoSaveCurTraceInfo {
|
||||
Message& mMsg;
|
||||
uint64_t mTaskId;
|
||||
uint64_t mSourceEventId;
|
||||
public:
|
||||
explicit AutoTaskTracerRun(Message& aMsg);
|
||||
~AutoTaskTracerRun();
|
||||
};
|
||||
#endif
|
||||
|
||||
#if !defined(OS_MACOSX)
|
||||
protected:
|
||||
#endif
|
||||
|
@ -311,6 +324,7 @@ class Message : public Pickle {
|
|||
// Sequence number
|
||||
int32_t seqno;
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
uint64_t task_id;
|
||||
uint64_t source_event_id;
|
||||
uint64_t parent_task_id;
|
||||
mozilla::tasktracer::SourceEventType source_event_type;
|
||||
|
|
|
@ -22,6 +22,11 @@
|
|||
#include "nsISupportsImpl.h"
|
||||
#include "nsContentUtils.h"
|
||||
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
#include "GeckoTaskTracer.h"
|
||||
using namespace mozilla::tasktracer;
|
||||
#endif
|
||||
|
||||
using mozilla::Move;
|
||||
|
||||
// Undo the damage done by mozzconf.h
|
||||
|
@ -979,6 +984,9 @@ MessageChannel::OnMessageReceivedFromLink(Message&& aMsg)
|
|||
// blocked. This is okay, since we always check for pending events before
|
||||
// blocking again.
|
||||
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
aMsg.TaskTracerDispatch();
|
||||
#endif
|
||||
RefPtr<MessageTask> task = new MessageTask(this, Move(aMsg));
|
||||
mPending.insertBack(task);
|
||||
|
||||
|
@ -1079,6 +1087,9 @@ MessageChannel::Send(Message* aMsg, Message* aReply)
|
|||
SyncStackFrame frame(this, false);
|
||||
NeuteredWindowRegion neuteredRgn(mFlags & REQUIRE_DEFERRED_MESSAGE_PROTECTION);
|
||||
#endif
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
AutoScopedLabel autolabel("sync message %s", aMsg->name());
|
||||
#endif
|
||||
|
||||
CxxStackFrame f(*this, OUT_MESSAGE, msg);
|
||||
|
||||
|
@ -1271,6 +1282,9 @@ MessageChannel::Call(Message* aMsg, Message* aReply)
|
|||
#ifdef OS_WIN
|
||||
SyncStackFrame frame(this, true);
|
||||
#endif
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
AutoScopedLabel autolabel("sync message %s", aMsg->name());
|
||||
#endif
|
||||
|
||||
// This must come before MonitorAutoLock, as its destructor acquires the
|
||||
// monitor lock.
|
||||
|
@ -1416,6 +1430,9 @@ MessageChannel::Call(Message* aMsg, Message* aReply)
|
|||
// own the monitor.
|
||||
size_t stackDepth = InterruptStackDepth();
|
||||
{
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
Message::AutoTaskTracerRun tasktracerRun(recvd);
|
||||
#endif
|
||||
MonitorAutoUnlock unlock(*mMonitor);
|
||||
|
||||
CxxStackFrame frame(*this, IN_MESSAGE, &recvd);
|
||||
|
@ -1675,6 +1692,9 @@ MessageChannel::DispatchMessage(Message &&aMsg)
|
|||
MOZ_RELEASE_ASSERT(!aMsg.is_sync() || id == transaction.TransactionID());
|
||||
|
||||
{
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
Message::AutoTaskTracerRun tasktracerRun(aMsg);
|
||||
#endif
|
||||
MonitorAutoUnlock unlock(*mMonitor);
|
||||
CxxStackFrame frame(*this, IN_MESSAGE, &aMsg);
|
||||
|
||||
|
@ -1711,6 +1731,9 @@ MessageChannel::DispatchSyncMessage(const Message& aMsg, Message*& aReply)
|
|||
int nestedLevel = aMsg.nested_level();
|
||||
|
||||
MOZ_RELEASE_ASSERT(nestedLevel == IPC::Message::NOT_NESTED || NS_IsMainThread());
|
||||
#ifdef MOZ_TASK_TRACER
|
||||
AutoScopedLabel autolabel("sync message %s", aMsg.name());
|
||||
#endif
|
||||
|
||||
MessageChannel* dummy;
|
||||
MessageChannel*& blockingVar = mSide == ChildSide && NS_IsMainThread() ? gParentProcessBlocker : dummy;
|
||||
|
|
|
@ -134,13 +134,13 @@ class GCRekeyableHashMap : public JS::GCHashMap<Key, Value, HashPolicy, AllocPol
|
|||
}
|
||||
};
|
||||
|
||||
template <typename Outer, typename... Args>
|
||||
class GCHashMapOperations
|
||||
template <typename Wrapper, typename... Args>
|
||||
class WrappedPtrOperations<JS::GCHashMap<Args...>, Wrapper>
|
||||
{
|
||||
using Map = JS::GCHashMap<Args...>;
|
||||
using Lookup = typename Map::Lookup;
|
||||
|
||||
const Map& map() const { return static_cast<const Outer*>(this)->get(); }
|
||||
const Map& map() const { return static_cast<const Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
using AddPtr = typename Map::AddPtr;
|
||||
|
@ -163,18 +163,18 @@ class GCHashMapOperations
|
|||
}
|
||||
};
|
||||
|
||||
template <typename Outer, typename... Args>
|
||||
class MutableGCHashMapOperations
|
||||
: public GCHashMapOperations<Outer, Args...>
|
||||
template <typename Wrapper, typename... Args>
|
||||
class MutableWrappedPtrOperations<JS::GCHashMap<Args...>, Wrapper>
|
||||
: public WrappedPtrOperations<JS::GCHashMap<Args...>, Wrapper>
|
||||
{
|
||||
using Map = JS::GCHashMap<Args...>;
|
||||
using Lookup = typename Map::Lookup;
|
||||
|
||||
Map& map() { return static_cast<Outer*>(this)->get(); }
|
||||
Map& map() { return static_cast<Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
using AddPtr = typename Map::AddPtr;
|
||||
struct Enum : public Map::Enum { explicit Enum(Outer& o) : Map::Enum(o.map()) {} };
|
||||
struct Enum : public Map::Enum { explicit Enum(Wrapper& o) : Map::Enum(o.map()) {} };
|
||||
using Ptr = typename Map::Ptr;
|
||||
using Range = typename Map::Range;
|
||||
|
||||
|
@ -211,26 +211,6 @@ class MutableGCHashMapOperations
|
|||
}
|
||||
};
|
||||
|
||||
template <typename A, typename B, typename C, typename D, typename E>
|
||||
class RootedBase<JS::GCHashMap<A,B,C,D,E>>
|
||||
: public MutableGCHashMapOperations<JS::Rooted<JS::GCHashMap<A,B,C,D,E>>, A,B,C,D,E>
|
||||
{};
|
||||
|
||||
template <typename A, typename B, typename C, typename D, typename E>
|
||||
class MutableHandleBase<JS::GCHashMap<A,B,C,D,E>>
|
||||
: public MutableGCHashMapOperations<JS::MutableHandle<JS::GCHashMap<A,B,C,D,E>>, A,B,C,D,E>
|
||||
{};
|
||||
|
||||
template <typename A, typename B, typename C, typename D, typename E>
|
||||
class HandleBase<JS::GCHashMap<A,B,C,D,E>>
|
||||
: public GCHashMapOperations<JS::Handle<JS::GCHashMap<A,B,C,D,E>>, A,B,C,D,E>
|
||||
{};
|
||||
|
||||
template <typename A, typename B, typename C, typename D, typename E>
|
||||
class WeakCacheBase<JS::GCHashMap<A,B,C,D,E>>
|
||||
: public MutableGCHashMapOperations<JS::WeakCache<JS::GCHashMap<A,B,C,D,E>>, A,B,C,D,E>
|
||||
{};
|
||||
|
||||
} // namespace js
|
||||
|
||||
namespace JS {
|
||||
|
@ -292,13 +272,13 @@ class GCHashSet : public js::HashSet<T, HashPolicy, AllocPolicy>
|
|||
|
||||
namespace js {
|
||||
|
||||
template <typename Outer, typename... Args>
|
||||
class GCHashSetOperations
|
||||
template <typename Wrapper, typename... Args>
|
||||
class WrappedPtrOperations<JS::GCHashSet<Args...>, Wrapper>
|
||||
{
|
||||
using Set = JS::GCHashSet<Args...>;
|
||||
using Lookup = typename Set::Lookup;
|
||||
|
||||
const Set& set() const { return static_cast<const Outer*>(this)->get(); }
|
||||
const Set& set() const { return static_cast<const Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
using AddPtr = typename Set::AddPtr;
|
||||
|
@ -322,19 +302,19 @@ class GCHashSetOperations
|
|||
}
|
||||
};
|
||||
|
||||
template <typename Outer, typename... Args>
|
||||
class MutableGCHashSetOperations
|
||||
: public GCHashSetOperations<Outer, Args...>
|
||||
template <typename Wrapper, typename... Args>
|
||||
class MutableWrappedPtrOperations<JS::GCHashSet<Args...>, Wrapper>
|
||||
: public WrappedPtrOperations<JS::GCHashSet<Args...>, Wrapper>
|
||||
{
|
||||
using Set = JS::GCHashSet<Args...>;
|
||||
using Lookup = typename Set::Lookup;
|
||||
|
||||
Set& set() { return static_cast<Outer*>(this)->get(); }
|
||||
Set& set() { return static_cast<Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
using AddPtr = typename Set::AddPtr;
|
||||
using Entry = typename Set::Entry;
|
||||
struct Enum : public Set::Enum { explicit Enum(Outer& o) : Set::Enum(o.set()) {} };
|
||||
struct Enum : public Set::Enum { explicit Enum(Wrapper& o) : Set::Enum(o.set()) {} };
|
||||
using Ptr = typename Set::Ptr;
|
||||
using Range = typename Set::Range;
|
||||
|
||||
|
@ -370,30 +350,6 @@ class MutableGCHashSetOperations
|
|||
}
|
||||
};
|
||||
|
||||
template <typename T, typename HP, typename AP>
|
||||
class RootedBase<JS::GCHashSet<T, HP, AP>>
|
||||
: public MutableGCHashSetOperations<JS::Rooted<JS::GCHashSet<T, HP, AP>>, T, HP, AP>
|
||||
{
|
||||
};
|
||||
|
||||
template <typename T, typename HP, typename AP>
|
||||
class MutableHandleBase<JS::GCHashSet<T, HP, AP>>
|
||||
: public MutableGCHashSetOperations<JS::MutableHandle<JS::GCHashSet<T, HP, AP>>, T, HP, AP>
|
||||
{
|
||||
};
|
||||
|
||||
template <typename T, typename HP, typename AP>
|
||||
class HandleBase<JS::GCHashSet<T, HP, AP>>
|
||||
: public GCHashSetOperations<JS::Handle<JS::GCHashSet<T, HP, AP>>, T, HP, AP>
|
||||
{
|
||||
};
|
||||
|
||||
template <typename T, typename HP, typename AP>
|
||||
class WeakCacheBase<JS::GCHashSet<T, HP, AP>>
|
||||
: public MutableGCHashSetOperations<JS::WeakCache<JS::GCHashSet<T, HP, AP>>, T, HP, AP>
|
||||
{
|
||||
};
|
||||
|
||||
} /* namespace js */
|
||||
|
||||
#endif /* GCHashTable_h */
|
||||
|
|
|
@ -124,13 +124,13 @@ struct GCPolicy<mozilla::Variant<Ts...>>
|
|||
|
||||
namespace js {
|
||||
|
||||
template <typename Outer, typename... Ts>
|
||||
class GCVariantOperations
|
||||
template <typename Wrapper, typename... Ts>
|
||||
class WrappedPtrOperations<mozilla::Variant<Ts...>, Wrapper>
|
||||
{
|
||||
using Impl = JS::detail::GCVariantImplementation<Ts...>;
|
||||
using Variant = mozilla::Variant<Ts...>;
|
||||
|
||||
const Variant& variant() const { return static_cast<const Outer*>(this)->get(); }
|
||||
const Variant& variant() const { return static_cast<const Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
template <typename T>
|
||||
|
@ -150,15 +150,15 @@ class GCVariantOperations
|
|||
}
|
||||
};
|
||||
|
||||
template <typename Outer, typename... Ts>
|
||||
class MutableGCVariantOperations
|
||||
: public GCVariantOperations<Outer, Ts...>
|
||||
template <typename Wrapper, typename... Ts>
|
||||
class MutableWrappedPtrOperations<mozilla::Variant<Ts...>, Wrapper>
|
||||
: public WrappedPtrOperations<mozilla::Variant<Ts...>, Wrapper>
|
||||
{
|
||||
using Impl = JS::detail::GCVariantImplementation<Ts...>;
|
||||
using Variant = mozilla::Variant<Ts...>;
|
||||
|
||||
const Variant& variant() const { return static_cast<const Outer*>(this)->get(); }
|
||||
Variant& variant() { return static_cast<Outer*>(this)->get(); }
|
||||
const Variant& variant() const { return static_cast<const Wrapper*>(this)->get(); }
|
||||
Variant& variant() { return static_cast<Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
template <typename T>
|
||||
|
@ -173,26 +173,6 @@ class MutableGCVariantOperations
|
|||
}
|
||||
};
|
||||
|
||||
template <typename... Ts>
|
||||
class RootedBase<mozilla::Variant<Ts...>>
|
||||
: public MutableGCVariantOperations<JS::Rooted<mozilla::Variant<Ts...>>, Ts...>
|
||||
{ };
|
||||
|
||||
template <typename... Ts>
|
||||
class MutableHandleBase<mozilla::Variant<Ts...>>
|
||||
: public MutableGCVariantOperations<JS::MutableHandle<mozilla::Variant<Ts...>>, Ts...>
|
||||
{ };
|
||||
|
||||
template <typename... Ts>
|
||||
class HandleBase<mozilla::Variant<Ts...>>
|
||||
: public GCVariantOperations<JS::Handle<mozilla::Variant<Ts...>>, Ts...>
|
||||
{ };
|
||||
|
||||
template <typename... Ts>
|
||||
class PersistentRootedBase<mozilla::Variant<Ts...>>
|
||||
: public MutableGCVariantOperations<JS::PersistentRooted<mozilla::Variant<Ts...>>, Ts...>
|
||||
{ };
|
||||
|
||||
} // namespace js
|
||||
|
||||
#endif // js_GCVariant_h
|
||||
|
|
|
@ -135,11 +135,11 @@ class GCVector
|
|||
|
||||
namespace js {
|
||||
|
||||
template <typename Outer, typename T, size_t Capacity, typename AllocPolicy>
|
||||
class GCVectorOperations
|
||||
template <typename Wrapper, typename T, size_t Capacity, typename AllocPolicy>
|
||||
class WrappedPtrOperations<JS::GCVector<T, Capacity, AllocPolicy>, Wrapper>
|
||||
{
|
||||
using Vec = JS::GCVector<T, Capacity, AllocPolicy>;
|
||||
const Vec& vec() const { return static_cast<const Outer*>(this)->get(); }
|
||||
const Vec& vec() const { return static_cast<const Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
const AllocPolicy& allocPolicy() const { return vec().allocPolicy(); }
|
||||
|
@ -155,13 +155,13 @@ class GCVectorOperations
|
|||
}
|
||||
};
|
||||
|
||||
template <typename Outer, typename T, size_t Capacity, typename AllocPolicy>
|
||||
class MutableGCVectorOperations
|
||||
: public GCVectorOperations<Outer, T, Capacity, AllocPolicy>
|
||||
template <typename Wrapper, typename T, size_t Capacity, typename AllocPolicy>
|
||||
class MutableWrappedPtrOperations<JS::GCVector<T, Capacity, AllocPolicy>, Wrapper>
|
||||
: public WrappedPtrOperations<JS::GCVector<T, Capacity, AllocPolicy>, Wrapper>
|
||||
{
|
||||
using Vec = JS::GCVector<T, Capacity, AllocPolicy>;
|
||||
const Vec& vec() const { return static_cast<const Outer*>(this)->get(); }
|
||||
Vec& vec() { return static_cast<Outer*>(this)->get(); }
|
||||
const Vec& vec() const { return static_cast<const Wrapper*>(this)->get(); }
|
||||
Vec& vec() { return static_cast<Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
const AllocPolicy& allocPolicy() const { return vec().allocPolicy(); }
|
||||
|
@ -224,26 +224,6 @@ class MutableGCVectorOperations
|
|||
void erase(T* aBegin, T* aEnd) { vec().erase(aBegin, aEnd); }
|
||||
};
|
||||
|
||||
template <typename T, size_t N, typename AP>
|
||||
class RootedBase<JS::GCVector<T,N,AP>>
|
||||
: public MutableGCVectorOperations<JS::Rooted<JS::GCVector<T,N,AP>>, T,N,AP>
|
||||
{};
|
||||
|
||||
template <typename T, size_t N, typename AP>
|
||||
class MutableHandleBase<JS::GCVector<T,N,AP>>
|
||||
: public MutableGCVectorOperations<JS::MutableHandle<JS::GCVector<T,N,AP>>, T,N,AP>
|
||||
{};
|
||||
|
||||
template <typename T, size_t N, typename AP>
|
||||
class HandleBase<JS::GCVector<T,N,AP>>
|
||||
: public GCVectorOperations<JS::Handle<JS::GCVector<T,N,AP>>, T,N,AP>
|
||||
{};
|
||||
|
||||
template <typename T, size_t N, typename AP>
|
||||
class PersistentRootedBase<JS::GCVector<T,N,AP>>
|
||||
: public MutableGCVectorOperations<JS::PersistentRooted<JS::GCVector<T,N,AP>>, T,N,AP>
|
||||
{};
|
||||
|
||||
} // namespace js
|
||||
|
||||
#endif // js_GCVector_h
|
||||
|
|
|
@ -113,17 +113,23 @@ template <typename T>
|
|||
struct BarrierMethods {
|
||||
};
|
||||
|
||||
template <typename T>
|
||||
class RootedBase {};
|
||||
template <typename Element, typename Wrapper>
|
||||
class WrappedPtrOperations {};
|
||||
|
||||
template <typename T>
|
||||
class HandleBase {};
|
||||
template <typename Element, typename Wrapper>
|
||||
class MutableWrappedPtrOperations : public WrappedPtrOperations<Element, Wrapper> {};
|
||||
|
||||
template <typename T>
|
||||
class MutableHandleBase {};
|
||||
template <typename T, typename Wrapper>
|
||||
class RootedBase : public MutableWrappedPtrOperations<T, Wrapper> {};
|
||||
|
||||
template <typename T>
|
||||
class HeapBase {};
|
||||
template <typename T, typename Wrapper>
|
||||
class HandleBase : public WrappedPtrOperations<T, Wrapper> {};
|
||||
|
||||
template <typename T, typename Wrapper>
|
||||
class MutableHandleBase : public MutableWrappedPtrOperations<T, Wrapper> {};
|
||||
|
||||
template <typename T, typename Wrapper>
|
||||
class HeapBase : public MutableWrappedPtrOperations<T, Wrapper> {};
|
||||
|
||||
// Cannot use FOR_EACH_HEAP_ABLE_GC_POINTER_TYPE, as this would import too many macros into scope
|
||||
template <typename T> struct IsHeapConstructibleType { static constexpr bool value = false; };
|
||||
|
@ -133,8 +139,8 @@ FOR_EACH_PUBLIC_GC_POINTER_TYPE(DECLARE_IS_HEAP_CONSTRUCTIBLE_TYPE)
|
|||
FOR_EACH_PUBLIC_TAGGED_GC_POINTER_TYPE(DECLARE_IS_HEAP_CONSTRUCTIBLE_TYPE)
|
||||
#undef DECLARE_IS_HEAP_CONSTRUCTIBLE_TYPE
|
||||
|
||||
template <typename T>
|
||||
class PersistentRootedBase {};
|
||||
template <typename T, typename Wrapper>
|
||||
class PersistentRootedBase : public MutableWrappedPtrOperations<T, Wrapper> {};
|
||||
|
||||
static void* const ConstNullValue = nullptr;
|
||||
|
||||
|
@ -223,7 +229,7 @@ AssertGCThingIsNotAnObjectSubclass(js::gc::Cell* cell) {}
|
|||
* Type T must be a public GC pointer type.
|
||||
*/
|
||||
template <typename T>
|
||||
class Heap : public js::HeapBase<T>
|
||||
class Heap : public js::HeapBase<T, Heap<T>>
|
||||
{
|
||||
// Please note: this can actually also be used by nsXBLMaybeCompiled<T>, for legacy reasons.
|
||||
static_assert(js::IsHeapConstructibleType<T>::value,
|
||||
|
@ -362,7 +368,7 @@ ScriptIsMarkedGray(const Heap<JSScript*>& script)
|
|||
* - It is not possible to store flag bits in a Heap<T>.
|
||||
*/
|
||||
template <typename T>
|
||||
class TenuredHeap : public js::HeapBase<T>
|
||||
class TenuredHeap : public js::HeapBase<T, TenuredHeap<T>>
|
||||
{
|
||||
public:
|
||||
using ElementType = T;
|
||||
|
@ -445,7 +451,7 @@ class TenuredHeap : public js::HeapBase<T>
|
|||
* specialization, define a HandleBase<T> specialization containing them.
|
||||
*/
|
||||
template <typename T>
|
||||
class MOZ_NONHEAP_CLASS Handle : public js::HandleBase<T>
|
||||
class MOZ_NONHEAP_CLASS Handle : public js::HandleBase<T, Handle<T>>
|
||||
{
|
||||
friend class JS::MutableHandle<T>;
|
||||
|
||||
|
@ -535,7 +541,7 @@ class MOZ_NONHEAP_CLASS Handle : public js::HandleBase<T>
|
|||
* them.
|
||||
*/
|
||||
template <typename T>
|
||||
class MOZ_STACK_CLASS MutableHandle : public js::MutableHandleBase<T>
|
||||
class MOZ_STACK_CLASS MutableHandle : public js::MutableHandleBase<T, MutableHandle<T>>
|
||||
{
|
||||
public:
|
||||
using ElementType = T;
|
||||
|
@ -747,7 +753,7 @@ namespace JS {
|
|||
* specialization, define a RootedBase<T> specialization containing them.
|
||||
*/
|
||||
template <typename T>
|
||||
class MOZ_RAII Rooted : public js::RootedBase<T>
|
||||
class MOZ_RAII Rooted : public js::RootedBase<T, Rooted<T>>
|
||||
{
|
||||
inline void registerWithRootLists(js::RootedListHeads& roots) {
|
||||
this->stack = &roots[JS::MapTypeToRootKind<T>::kind];
|
||||
|
@ -845,8 +851,8 @@ namespace js {
|
|||
* Rooted<StringObject*> rooted(cx, &obj->as<StringObject*>());
|
||||
* Handle<StringObject*> h = rooted;
|
||||
*/
|
||||
template <>
|
||||
class RootedBase<JSObject*>
|
||||
template <typename Container>
|
||||
class RootedBase<JSObject*, Container> : public MutableWrappedPtrOperations<JSObject*, Container>
|
||||
{
|
||||
public:
|
||||
template <class U>
|
||||
|
@ -863,8 +869,8 @@ class RootedBase<JSObject*>
|
|||
* Rooted<StringObject*> rooted(cx, &obj->as<StringObject*>());
|
||||
* Handle<StringObject*> h = rooted;
|
||||
*/
|
||||
template <>
|
||||
class HandleBase<JSObject*>
|
||||
template <typename Container>
|
||||
class HandleBase<JSObject*, Container> : public WrappedPtrOperations<JSObject*, Container>
|
||||
{
|
||||
public:
|
||||
template <class U>
|
||||
|
@ -873,7 +879,7 @@ class HandleBase<JSObject*>
|
|||
|
||||
/** Interface substitute for Rooted<T> which does not root the variable's memory. */
|
||||
template <typename T>
|
||||
class MOZ_RAII FakeRooted : public RootedBase<T>
|
||||
class MOZ_RAII FakeRooted : public RootedBase<T, FakeRooted<T>>
|
||||
{
|
||||
public:
|
||||
using ElementType = T;
|
||||
|
@ -901,7 +907,7 @@ class MOZ_RAII FakeRooted : public RootedBase<T>
|
|||
|
||||
/** Interface substitute for MutableHandle<T> which is not required to point to rooted memory. */
|
||||
template <typename T>
|
||||
class FakeMutableHandle : public js::MutableHandleBase<T>
|
||||
class FakeMutableHandle : public js::MutableHandleBase<T, FakeMutableHandle<T>>
|
||||
{
|
||||
public:
|
||||
using ElementType = T;
|
||||
|
@ -1070,7 +1076,7 @@ MutableHandle<T>::MutableHandle(PersistentRooted<T>* root)
|
|||
* marked when the object itself is marked.
|
||||
*/
|
||||
template<typename T>
|
||||
class PersistentRooted : public js::PersistentRootedBase<T>,
|
||||
class PersistentRooted : public js::RootedBase<T, PersistentRooted<T>>,
|
||||
private mozilla::LinkedListElement<PersistentRooted<T>>
|
||||
{
|
||||
using ListBase = mozilla::LinkedListElement<PersistentRooted<T>>;
|
||||
|
@ -1230,44 +1236,25 @@ class JS_PUBLIC_API(ObjectPtr)
|
|||
|
||||
namespace js {
|
||||
|
||||
template <typename Outer, typename T, typename D>
|
||||
class UniquePtrOperations
|
||||
template <typename T, typename D, typename Container>
|
||||
class WrappedPtrOperations<UniquePtr<T, D>, Container>
|
||||
{
|
||||
const UniquePtr<T, D>& uniquePtr() const { return static_cast<const Outer*>(this)->get(); }
|
||||
const UniquePtr<T, D>& uniquePtr() const { return static_cast<const Container*>(this)->get(); }
|
||||
|
||||
public:
|
||||
explicit operator bool() const { return !!uniquePtr(); }
|
||||
};
|
||||
|
||||
template <typename Outer, typename T, typename D>
|
||||
class MutableUniquePtrOperations : public UniquePtrOperations<Outer, T, D>
|
||||
template <typename T, typename D, typename Container>
|
||||
class MutableWrappedPtrOperations<UniquePtr<T, D>, Container>
|
||||
: public WrappedPtrOperations<UniquePtr<T, D>, Container>
|
||||
{
|
||||
UniquePtr<T, D>& uniquePtr() { return static_cast<Outer*>(this)->get(); }
|
||||
UniquePtr<T, D>& uniquePtr() { return static_cast<Container*>(this)->get(); }
|
||||
|
||||
public:
|
||||
MOZ_MUST_USE typename UniquePtr<T, D>::Pointer release() { return uniquePtr().release(); }
|
||||
};
|
||||
|
||||
template <typename T, typename D>
|
||||
class RootedBase<UniquePtr<T, D>>
|
||||
: public MutableUniquePtrOperations<JS::Rooted<UniquePtr<T, D>>, T, D>
|
||||
{ };
|
||||
|
||||
template <typename T, typename D>
|
||||
class MutableHandleBase<UniquePtr<T, D>>
|
||||
: public MutableUniquePtrOperations<JS::MutableHandle<UniquePtr<T, D>>, T, D>
|
||||
{ };
|
||||
|
||||
template <typename T, typename D>
|
||||
class HandleBase<UniquePtr<T, D>>
|
||||
: public UniquePtrOperations<JS::Handle<UniquePtr<T, D>>, T, D>
|
||||
{ };
|
||||
|
||||
template <typename T, typename D>
|
||||
class PersistentRootedBase<UniquePtr<T, D>>
|
||||
: public MutableUniquePtrOperations<JS::PersistentRooted<UniquePtr<T, D>>, T, D>
|
||||
{ };
|
||||
|
||||
namespace gc {
|
||||
|
||||
template <typename T, typename TraceCallbacks>
|
||||
|
|
|
@ -9,11 +9,6 @@
|
|||
|
||||
#include "js/HeapAPI.h"
|
||||
|
||||
namespace js {
|
||||
template <typename T>
|
||||
class WeakCacheBase {};
|
||||
} // namespace js
|
||||
|
||||
namespace JS {
|
||||
template <typename T> class WeakCache;
|
||||
|
||||
|
@ -25,7 +20,7 @@ RegisterWeakCache(JS::Zone* zone, JS::WeakCache<void*>* cachep);
|
|||
// A WeakCache stores the given Sweepable container and links itself into a
|
||||
// list of such caches that are swept during each GC.
|
||||
template <typename T>
|
||||
class WeakCache : public js::WeakCacheBase<T>,
|
||||
class WeakCache : public js::MutableWrappedPtrOperations<T, WeakCache<T>>,
|
||||
private mozilla::LinkedListElement<WeakCache<T>>
|
||||
{
|
||||
friend class mozilla::LinkedListElement<WeakCache<T>>;
|
||||
|
|
|
@ -1288,20 +1288,18 @@ struct BarrierMethods<JS::Value>
|
|||
}
|
||||
};
|
||||
|
||||
template <class Outer> class MutableValueOperations;
|
||||
template <class Wrapper> class MutableValueOperations;
|
||||
|
||||
/**
|
||||
* A class designed for CRTP use in implementing the non-mutating parts of the
|
||||
* Value interface in Value-like classes. Outer must be a class inheriting
|
||||
* ValueOperations<Outer> with a visible get() method returning a const
|
||||
* reference to the Value abstracted by Outer.
|
||||
* Value interface in Value-like classes. Wrapper must be a class inheriting
|
||||
* ValueOperations<Wrapper> with a visible get() method returning a const
|
||||
* reference to the Value abstracted by Wrapper.
|
||||
*/
|
||||
template <class Outer>
|
||||
class ValueOperations
|
||||
template <class Wrapper>
|
||||
class WrappedPtrOperations<JS::Value, Wrapper>
|
||||
{
|
||||
friend class MutableValueOperations<Outer>;
|
||||
|
||||
const JS::Value& value() const { return static_cast<const Outer*>(this)->get(); }
|
||||
const JS::Value& value() const { return static_cast<const Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
bool isUndefined() const { return value().isUndefined(); }
|
||||
|
@ -1346,14 +1344,14 @@ class ValueOperations
|
|||
|
||||
/**
|
||||
* A class designed for CRTP use in implementing all the mutating parts of the
|
||||
* Value interface in Value-like classes. Outer must be a class inheriting
|
||||
* MutableValueOperations<Outer> with visible get() methods returning const and
|
||||
* non-const references to the Value abstracted by Outer.
|
||||
* Value interface in Value-like classes. Wrapper must be a class inheriting
|
||||
* MutableWrappedPtrOperations<Wrapper> with visible get() methods returning const and
|
||||
* non-const references to the Value abstracted by Wrapper.
|
||||
*/
|
||||
template <class Outer>
|
||||
class MutableValueOperations : public ValueOperations<Outer>
|
||||
template <class Wrapper>
|
||||
class MutableWrappedPtrOperations<JS::Value, Wrapper> : public WrappedPtrOperations<JS::Value, Wrapper>
|
||||
{
|
||||
JS::Value& value() { return static_cast<Outer*>(this)->get(); }
|
||||
JS::Value& value() { return static_cast<Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
void setNull() { value().setNull(); }
|
||||
|
@ -1378,13 +1376,9 @@ class MutableValueOperations : public ValueOperations<Outer>
|
|||
* Augment the generic Heap<T> interface when T = Value with
|
||||
* type-querying, value-extracting, and mutating operations.
|
||||
*/
|
||||
template <>
|
||||
class HeapBase<JS::Value> : public ValueOperations<JS::Heap<JS::Value> >
|
||||
template <typename Wrapper>
|
||||
class HeapBase<JS::Value, Wrapper> : public WrappedPtrOperations<JS::Value, Wrapper>
|
||||
{
|
||||
typedef JS::Heap<JS::Value> Outer;
|
||||
|
||||
friend class ValueOperations<Outer>;
|
||||
|
||||
void setBarriered(const JS::Value& v) {
|
||||
*static_cast<JS::Heap<JS::Value>*>(this) = v;
|
||||
}
|
||||
|
@ -1431,22 +1425,6 @@ class HeapBase<JS::Value> : public ValueOperations<JS::Heap<JS::Value> >
|
|||
}
|
||||
};
|
||||
|
||||
template <>
|
||||
class HandleBase<JS::Value> : public ValueOperations<JS::Handle<JS::Value> >
|
||||
{};
|
||||
|
||||
template <>
|
||||
class MutableHandleBase<JS::Value> : public MutableValueOperations<JS::MutableHandle<JS::Value> >
|
||||
{};
|
||||
|
||||
template <>
|
||||
class RootedBase<JS::Value> : public MutableValueOperations<JS::Rooted<JS::Value> >
|
||||
{};
|
||||
|
||||
template <>
|
||||
class PersistentRootedBase<JS::Value> : public MutableValueOperations<JS::PersistentRooted<JS::Value>>
|
||||
{};
|
||||
|
||||
/*
|
||||
* If the Value is a GC pointer type, convert to that type and call |f| with
|
||||
* the pointer. If the Value is not a GC type, calls F::defaultValue.
|
||||
|
|
|
@ -53,14 +53,22 @@ class HashableValue
|
|||
}
|
||||
};
|
||||
|
||||
template <>
|
||||
class RootedBase<HashableValue> {
|
||||
template <typename Wrapper>
|
||||
class WrappedPtrOperations<HashableValue, Wrapper>
|
||||
{
|
||||
public:
|
||||
Value value() const {
|
||||
return static_cast<const Wrapper*>(this)->get().get();
|
||||
}
|
||||
};
|
||||
|
||||
template <typename Wrapper>
|
||||
class MutableWrappedPtrOperations<HashableValue, Wrapper>
|
||||
: public WrappedPtrOperations<HashableValue, Wrapper>
|
||||
{
|
||||
public:
|
||||
MOZ_MUST_USE bool setValue(JSContext* cx, HandleValue v) {
|
||||
return static_cast<JS::Rooted<HashableValue>*>(this)->get().setValue(cx, v);
|
||||
}
|
||||
Value value() const {
|
||||
return static_cast<const JS::Rooted<HashableValue>*>(this)->get().get();
|
||||
return static_cast<Wrapper*>(this)->get().setValue(cx, v);
|
||||
}
|
||||
};
|
||||
|
||||
|
|
|
@ -786,7 +786,17 @@ ScheduleGC(JSContext* cx, unsigned argc, Value* vp)
|
|||
PrepareZoneForGC(zone);
|
||||
} else if (args[0].isString()) {
|
||||
/* This allows us to schedule the atoms zone for GC. */
|
||||
PrepareZoneForGC(args[0].toString()->zone());
|
||||
Zone* zone = args[0].toString()->zoneFromAnyThread();
|
||||
if (!CurrentThreadCanAccessZone(zone)) {
|
||||
RootedObject callee(cx, &args.callee());
|
||||
ReportUsageErrorASCII(cx, callee, "Specified zone not accessible for GC");
|
||||
return false;
|
||||
}
|
||||
PrepareZoneForGC(zone);
|
||||
} else {
|
||||
RootedObject callee(cx, &args.callee());
|
||||
ReportUsageErrorASCII(cx, callee, "Bad argument - expecting integer, object or string");
|
||||
return false;
|
||||
}
|
||||
|
||||
uint32_t zealBits;
|
||||
|
@ -4219,9 +4229,10 @@ JS_FN_HELP("rejectPromise", RejectPromise, 2, 0,
|
|||
gc::ZealModeHelpText),
|
||||
|
||||
JS_FN_HELP("schedulegc", ScheduleGC, 1, 0,
|
||||
"schedulegc([num | obj])",
|
||||
"schedulegc([num | obj | string])",
|
||||
" If num is given, schedule a GC after num allocations.\n"
|
||||
" If obj is given, schedule a GC of obj's zone.\n"
|
||||
" If string is given, schedule a GC of the string's zone if possible.\n"
|
||||
" Returns the number of allocations before the next trigger."),
|
||||
|
||||
JS_FN_HELP("selectforgc", SelectForGC, 0, 0,
|
||||
|
|
|
@ -52,11 +52,11 @@ class TraceableFifo : public js::Fifo<T, MinInlineCapacity, AllocPolicy>
|
|||
}
|
||||
};
|
||||
|
||||
template <typename Outer, typename T, size_t Capacity, typename AllocPolicy>
|
||||
class TraceableFifoOperations
|
||||
template <typename Wrapper, typename T, size_t Capacity, typename AllocPolicy>
|
||||
class WrappedPtrOperations<TraceableFifo<T, Capacity, AllocPolicy>, Wrapper>
|
||||
{
|
||||
using TF = TraceableFifo<T, Capacity, AllocPolicy>;
|
||||
const TF& fifo() const { return static_cast<const Outer*>(this)->extract(); }
|
||||
const TF& fifo() const { return static_cast<const Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
size_t length() const { return fifo().length(); }
|
||||
|
@ -64,12 +64,12 @@ class TraceableFifoOperations
|
|||
const T& front() const { return fifo().front(); }
|
||||
};
|
||||
|
||||
template <typename Outer, typename T, size_t Capacity, typename AllocPolicy>
|
||||
class MutableTraceableFifoOperations
|
||||
: public TraceableFifoOperations<Outer, T, Capacity, AllocPolicy>
|
||||
template <typename Wrapper, typename T, size_t Capacity, typename AllocPolicy>
|
||||
class MutableWrappedPtrOperations<TraceableFifo<T, Capacity, AllocPolicy>, Wrapper>
|
||||
: public WrappedPtrOperations<TraceableFifo<T, Capacity, AllocPolicy>, Wrapper>
|
||||
{
|
||||
using TF = TraceableFifo<T, Capacity, AllocPolicy>;
|
||||
TF& fifo() { return static_cast<Outer*>(this)->extract(); }
|
||||
TF& fifo() { return static_cast<Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
T& front() { return fifo().front(); }
|
||||
|
@ -83,46 +83,6 @@ class MutableTraceableFifoOperations
|
|||
void clear() { fifo().clear(); }
|
||||
};
|
||||
|
||||
template <typename A, size_t B, typename C>
|
||||
class RootedBase<TraceableFifo<A,B,C>>
|
||||
: public MutableTraceableFifoOperations<JS::Rooted<TraceableFifo<A,B,C>>, A,B,C>
|
||||
{
|
||||
using TF = TraceableFifo<A,B,C>;
|
||||
|
||||
friend class TraceableFifoOperations<JS::Rooted<TF>, A,B,C>;
|
||||
const TF& extract() const { return *static_cast<const JS::Rooted<TF>*>(this)->address(); }
|
||||
|
||||
friend class MutableTraceableFifoOperations<JS::Rooted<TF>, A,B,C>;
|
||||
TF& extract() { return *static_cast<JS::Rooted<TF>*>(this)->address(); }
|
||||
};
|
||||
|
||||
template <typename A, size_t B, typename C>
|
||||
class MutableHandleBase<TraceableFifo<A,B,C>>
|
||||
: public MutableTraceableFifoOperations<JS::MutableHandle<TraceableFifo<A,B,C>>, A,B,C>
|
||||
{
|
||||
using TF = TraceableFifo<A,B,C>;
|
||||
|
||||
friend class TraceableFifoOperations<JS::MutableHandle<TF>, A,B,C>;
|
||||
const TF& extract() const {
|
||||
return *static_cast<const JS::MutableHandle<TF>*>(this)->address();
|
||||
}
|
||||
|
||||
friend class MutableTraceableFifoOperations<JS::MutableHandle<TF>, A,B,C>;
|
||||
TF& extract() { return *static_cast<JS::MutableHandle<TF>*>(this)->address(); }
|
||||
};
|
||||
|
||||
template <typename A, size_t B, typename C>
|
||||
class HandleBase<TraceableFifo<A,B,C>>
|
||||
: public TraceableFifoOperations<JS::Handle<TraceableFifo<A,B,C>>, A,B,C>
|
||||
{
|
||||
using TF = TraceableFifo<A,B,C>;
|
||||
|
||||
friend class TraceableFifoOperations<JS::Handle<TF>, A,B,C>;
|
||||
const TF& extract() const {
|
||||
return *static_cast<const JS::Handle<TF>*>(this)->address();
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace js
|
||||
|
||||
#endif // js_TraceableFifo_h
|
||||
|
|
|
@ -316,15 +316,9 @@ struct InternalBarrierMethods<jsid>
|
|||
static void postBarrier(jsid* idp, jsid prev, jsid next) {}
|
||||
};
|
||||
|
||||
// Barrier classes can use Mixins to add methods to a set of barrier
|
||||
// instantiations, to make the barriered thing look and feel more like the
|
||||
// thing itself.
|
||||
template <typename T>
|
||||
class BarrieredBaseMixins {};
|
||||
|
||||
// Base class of all barrier types.
|
||||
template <typename T>
|
||||
class BarrieredBase : public BarrieredBaseMixins<T>
|
||||
class BarrieredBase
|
||||
{
|
||||
protected:
|
||||
// BarrieredBase is not directly instantiable.
|
||||
|
@ -345,9 +339,12 @@ class BarrieredBase : public BarrieredBaseMixins<T>
|
|||
|
||||
// Base class for barriered pointer types that intercept only writes.
|
||||
template <class T>
|
||||
class WriteBarrieredBase : public BarrieredBase<T>
|
||||
class WriteBarrieredBase : public BarrieredBase<T>,
|
||||
public WrappedPtrOperations<T, WriteBarrieredBase<T>>
|
||||
{
|
||||
protected:
|
||||
using BarrieredBase<T>::value;
|
||||
|
||||
// WriteBarrieredBase is not directly instantiable.
|
||||
explicit WriteBarrieredBase(const T& v) : BarrieredBase<T>(v) {}
|
||||
|
||||
|
@ -566,8 +563,12 @@ class ReadBarrieredBase : public BarrieredBase<T>
|
|||
// insert manual post-barriers on the table for rekeying if the key is based in
|
||||
// any way on the address of the object.
|
||||
template <typename T>
|
||||
class ReadBarriered : public ReadBarrieredBase<T>
|
||||
class ReadBarriered : public ReadBarrieredBase<T>,
|
||||
public WrappedPtrOperations<T, ReadBarriered<T>>
|
||||
{
|
||||
protected:
|
||||
using ReadBarrieredBase<T>::value;
|
||||
|
||||
public:
|
||||
ReadBarriered() : ReadBarrieredBase<T>(JS::GCPolicy<T>::initial()) {}
|
||||
|
||||
|
@ -634,12 +635,6 @@ class ReadBarriered : public ReadBarrieredBase<T>
|
|||
template <typename T>
|
||||
using WeakRef = ReadBarriered<T>;
|
||||
|
||||
// Add Value operations to all Barrier types. Note, this must be defined before
|
||||
// HeapSlot for HeapSlot's base to get these operations.
|
||||
template <>
|
||||
class BarrieredBaseMixins<JS::Value> : public ValueOperations<WriteBarrieredBase<JS::Value>>
|
||||
{};
|
||||
|
||||
// A pre- and post-barriered Value that is specialized to be aware that it
|
||||
// resides in a slots or elements vector. This allows it to be relocated in
|
||||
// memory, but with substantially less overhead than a HeapPtr.
|
||||
|
|
|
@ -0,0 +1,3 @@
|
|||
if (helperThreadCount() === 0)
|
||||
quit();
|
||||
evalInWorker(`schedulegc("s1");`);
|
|
@ -0,0 +1,15 @@
|
|||
if (!('gczeal' in this))
|
||||
quit();
|
||||
var lfGlobal = newGlobal();
|
||||
lfGlobal.evaluate(`
|
||||
for (var i = 0; i < 600; i++)
|
||||
eval('function f' + i + '() {}');
|
||||
`);
|
||||
var lfGlobal = newGlobal();
|
||||
lfGlobal.evaluate(`
|
||||
if (!('gczeal' in this))
|
||||
quit();
|
||||
var dbg = new Debugger();
|
||||
gczeal(9, 10);
|
||||
dbg.findScripts({});
|
||||
`);
|
|
@ -0,0 +1,15 @@
|
|||
if (!('gczeal' in this))
|
||||
quit();
|
||||
var lfGlobal = newGlobal();
|
||||
lfGlobal.evaluate(`
|
||||
for (var i = 0; i < 600; i++)
|
||||
eval('function f' + i + '() {}');
|
||||
`);
|
||||
var lfGlobal = newGlobal();
|
||||
lfGlobal.evaluate(`
|
||||
if (!('gczeal' in this))
|
||||
quit();
|
||||
var dbg = new Debugger();
|
||||
gczeal(9, 1);
|
||||
dbg.findScripts({});
|
||||
`);
|
|
@ -207,33 +207,68 @@ assertEq(testTrunc(13.37), 1);
|
|||
testTrap64('rem_s', 10, 0, /integer divide by zero/);
|
||||
testTrap64('rem_u', 10, 0, /integer divide by zero/);
|
||||
|
||||
testBinary64('and', 42, 6, 2);
|
||||
// Bitops.
|
||||
testBinary64('or', 42, 6, 46);
|
||||
testBinary64('xor', 42, 2, 40);
|
||||
testBinary64('and', "0x8765432112345678", "0xffff0000ffff0000", "0x8765000012340000");
|
||||
testBinary64('or', "0x8765432112345678", "0xffff0000ffff0000", "0xffff4321ffff5678");
|
||||
|
||||
testBinary64('xor', 42, 2, 40);
|
||||
testBinary64('xor', "0x8765432112345678", "0xffff0000ffff0000", "0x789a4321edcb5678");
|
||||
testBinary64('shl', 40, 2, 160);
|
||||
testBinary64('shr_s', -40, 2, -10);
|
||||
testBinary64('shr_u', -40, 2, "0x3ffffffffffffff6");
|
||||
testBinary64('shl', 0xff00ff, 28, "0xff00ff0000000");
|
||||
|
||||
testBinary64('shl', 0xff00ff, 28, "0x0ff00ff0000000");
|
||||
testBinary64('shl', 0xff00ff, 30, "0x3fc03fc0000000");
|
||||
testBinary64('shl', 0xff00ff, 31, "0x7f807f80000000");
|
||||
testBinary64('shl', 0xff00ff, 32, "0xff00ff00000000");
|
||||
testBinary64('shl', 1, 63, "0x8000000000000000");
|
||||
testBinary64('shl', 1, 64, 1);
|
||||
testBinary64('shr_s', "0xff00ff0000000", 28, 0xff00ff);
|
||||
testBinary64('shr_u', "0x8ffff00ff0000000", 56, 0x8f);
|
||||
testBinary64('rotl', 40, 2, 160);
|
||||
testBinary64('rotr', 40, 2, 10);
|
||||
testBinary64('rotr', "0x1234567812345678", 4, "0x8123456781234567");
|
||||
testBinary64('rotl', "0x1234567812345678", 4, "0x2345678123456781");
|
||||
testBinary64('rotl', "0x1234567812345678", 60, "0x8123456781234567");
|
||||
testBinary64('rotr', "0x1234567812345678", 60, "0x2345678123456781");
|
||||
testBinary64('rotl', "0x0000000000001000", 127, "0x0000000000000800");
|
||||
testBinary64('rotr', "0x0000000000001000", 127, "0x0000000000002000");
|
||||
testBinary64('rotr', 40, 0, 40);
|
||||
testBinary64('rotl', 40, 0, 40);
|
||||
testBinary64('and', 42, 0, 0);
|
||||
testBinary64('and', "0x0000000012345678", "0xffff0000ffff0000", "0x0000000012340000");
|
||||
testBinary64('shl', 40, 2, 160);
|
||||
|
||||
testBinary64('shr_s', -40, 2, -10);
|
||||
testBinary64('shr_s', "0xff00ff0000000", 28, 0xff00ff);
|
||||
testBinary64('shr_s', "0xff00ff0000000", 30, 0x3fc03f);
|
||||
testBinary64('shr_s', "0xff00ff0000000", 31, 0x1fe01f);
|
||||
testBinary64('shr_s', "0xff00ff0000000", 32, 0x0ff00f);
|
||||
|
||||
testBinary64('shr_u', -40, 2, "0x3ffffffffffffff6");
|
||||
testBinary64('shr_u', "0x8ffff00ff0000000", 30, "0x23fffc03f");
|
||||
testBinary64('shr_u', "0x8ffff00ff0000000", 31, "0x11fffe01f");
|
||||
testBinary64('shr_u', "0x8ffff00ff0000000", 32, "0x08ffff00f");
|
||||
testBinary64('shr_u', "0x8ffff00ff0000000", 56, 0x8f);
|
||||
|
||||
testBinary64('and', 42, 0, 0);
|
||||
testBinary64('and', 42, 6, 2);
|
||||
testBinary64('and', "0x0000000012345678", "0xffff0000ffff0000", "0x0000000012340000");
|
||||
testBinary64('and', "0x8765432112345678", "0xffff0000ffff0000", "0x8765000012340000");
|
||||
|
||||
// Rotations.
|
||||
testBinary64('rotl', 40, 0, 0x28);
|
||||
testBinary64('rotl', 40, 2, 0xA0);
|
||||
testBinary64('rotl', 40, 8, 0x2800);
|
||||
testBinary64('rotl', 40, 30, "0xA00000000");
|
||||
testBinary64('rotl', 40, 31, "0x1400000000");
|
||||
testBinary64('rotl', 40, 32, "0x2800000000");
|
||||
|
||||
testBinary64('rotl', "0x1234567812345678", 4, "0x2345678123456781");
|
||||
testBinary64('rotl', "0x1234567812345678", 30, "0x048D159E048D159E");
|
||||
testBinary64('rotl', "0x1234567812345678", 31, "0x091A2B3C091A2B3C");
|
||||
testBinary64('rotl', "0x1234567812345678", 32, "0x1234567812345678");
|
||||
|
||||
testBinary64('rotl', "0x0000000000001000", 127, "0x0000000000000800");
|
||||
|
||||
testBinary64('rotr', 40, 0, 0x28);
|
||||
testBinary64('rotr', 40, 2, 0x0A);
|
||||
testBinary64('rotr', 40, 30, "0xA000000000");
|
||||
testBinary64('rotr', 40, 31, "0x5000000000");
|
||||
testBinary64('rotr', 40, 32, "0x2800000000");
|
||||
|
||||
testBinary64('rotr', "0x1234567812345678", 4, "0x8123456781234567");
|
||||
testBinary64('rotr', "0x1234567812345678", 30, "0x48D159E048D159E0");
|
||||
testBinary64('rotr', "0x1234567812345678", 31, "0x2468ACF02468ACF0");
|
||||
testBinary64('rotr', "0x1234567812345678", 32, "0x1234567812345678");
|
||||
testBinary64('rotr', "0x1234567812345678", 60, "0x2345678123456781");
|
||||
|
||||
testBinary64('rotr', "0x0000000000001000", 127, "0x0000000000002000");
|
||||
|
||||
// Comparisons.
|
||||
testComparison64('eq', 40, 40, 1);
|
||||
testComparison64('ne', 40, 40, 0);
|
||||
testComparison64('lt_s', 40, 40, 0);
|
||||
|
|
|
@ -0,0 +1,9 @@
|
|||
if (typeof enableSPSProfiling === 'undefined' || !isAsmJSCompilationAvailable())
|
||||
quit();
|
||||
|
||||
enableSPSProfiling();
|
||||
var code = evaluate("(function() { 'use asm'; function g() { return 43 } return g })", {
|
||||
fileName: null
|
||||
});
|
||||
|
||||
assertEq(code()(), 43);
|
|
@ -4729,6 +4729,9 @@ IonBuilder::getSingletonPrototype(JSFunction* target)
|
|||
MDefinition*
|
||||
IonBuilder::createThisScriptedSingleton(JSFunction* target, MDefinition* callee)
|
||||
{
|
||||
if (!target->hasScript())
|
||||
return nullptr;
|
||||
|
||||
// Get the singleton prototype (if exists)
|
||||
JSObject* proto = getSingletonPrototype(target);
|
||||
if (!proto)
|
||||
|
|
|
@ -539,7 +539,7 @@ MacroAssembler::rotateLeft64(Imm32 count, Register64 src, Register64 dest, Regis
|
|||
MOZ_ASSERT(src == dest, "defineReuseInput");
|
||||
|
||||
int32_t amount = count.value & 0x3f;
|
||||
if (amount % 0x1f != 0) {
|
||||
if ((amount & 0x1f) != 0) {
|
||||
movl(dest.high, temp);
|
||||
shldl(Imm32(amount & 0x1f), dest.low, dest.high);
|
||||
shldl(Imm32(amount & 0x1f), temp, dest.low);
|
||||
|
|
|
@ -57,19 +57,10 @@ struct MyContainer
|
|||
};
|
||||
|
||||
namespace js {
|
||||
template <>
|
||||
struct RootedBase<MyContainer> {
|
||||
HeapPtr<JSObject*>& obj() { return static_cast<Rooted<MyContainer>*>(this)->get().obj; }
|
||||
HeapPtr<JSString*>& str() { return static_cast<Rooted<MyContainer>*>(this)->get().str; }
|
||||
};
|
||||
template <>
|
||||
struct PersistentRootedBase<MyContainer> {
|
||||
HeapPtr<JSObject*>& obj() {
|
||||
return static_cast<PersistentRooted<MyContainer>*>(this)->get().obj;
|
||||
}
|
||||
HeapPtr<JSString*>& str() {
|
||||
return static_cast<PersistentRooted<MyContainer>*>(this)->get().str;
|
||||
}
|
||||
template <typename Wrapper>
|
||||
struct MutableWrappedPtrOperations<MyContainer, Wrapper> {
|
||||
HeapPtr<JSObject*>& obj() { return static_cast<Wrapper*>(this)->get().obj; }
|
||||
HeapPtr<JSString*>& str() { return static_cast<Wrapper*>(this)->get().str; }
|
||||
};
|
||||
} // namespace js
|
||||
|
||||
|
|
|
@ -2569,10 +2569,14 @@ struct JS_PUBLIC_API(PropertyDescriptor) {
|
|||
void trace(JSTracer* trc);
|
||||
};
|
||||
|
||||
template <typename Outer>
|
||||
class PropertyDescriptorOperations
|
||||
} // namespace JS
|
||||
|
||||
namespace js {
|
||||
|
||||
template <typename Wrapper>
|
||||
class WrappedPtrOperations<JS::PropertyDescriptor, Wrapper>
|
||||
{
|
||||
const PropertyDescriptor& desc() const { return static_cast<const Outer*>(this)->get(); }
|
||||
const JS::PropertyDescriptor& desc() const { return static_cast<const Wrapper*>(this)->get(); }
|
||||
|
||||
bool has(unsigned bit) const {
|
||||
MOZ_ASSERT(bit != 0);
|
||||
|
@ -2701,10 +2705,11 @@ class PropertyDescriptorOperations
|
|||
}
|
||||
};
|
||||
|
||||
template <typename Outer>
|
||||
class MutablePropertyDescriptorOperations : public PropertyDescriptorOperations<Outer>
|
||||
template <typename Wrapper>
|
||||
class MutableWrappedPtrOperations<JS::PropertyDescriptor, Wrapper>
|
||||
: public js::WrappedPtrOperations<JS::PropertyDescriptor, Wrapper>
|
||||
{
|
||||
PropertyDescriptor& desc() { return static_cast<Outer*>(this)->get(); }
|
||||
JS::PropertyDescriptor& desc() { return static_cast<Wrapper*>(this)->get(); }
|
||||
|
||||
public:
|
||||
void clear() {
|
||||
|
@ -2715,7 +2720,7 @@ class MutablePropertyDescriptorOperations : public PropertyDescriptorOperations<
|
|||
value().setUndefined();
|
||||
}
|
||||
|
||||
void initFields(HandleObject obj, HandleValue v, unsigned attrs,
|
||||
void initFields(JS::HandleObject obj, JS::HandleValue v, unsigned attrs,
|
||||
JSGetterOp getterOp, JSSetterOp setterOp) {
|
||||
MOZ_ASSERT(getterOp != JS_PropertyStub);
|
||||
MOZ_ASSERT(setterOp != JS_StrictPropertyStub);
|
||||
|
@ -2727,7 +2732,7 @@ class MutablePropertyDescriptorOperations : public PropertyDescriptorOperations<
|
|||
setSetter(setterOp);
|
||||
}
|
||||
|
||||
void assign(PropertyDescriptor& other) {
|
||||
void assign(JS::PropertyDescriptor& other) {
|
||||
object().set(other.obj);
|
||||
setAttributes(other.attrs);
|
||||
setGetter(other.getter);
|
||||
|
@ -2735,7 +2740,7 @@ class MutablePropertyDescriptorOperations : public PropertyDescriptorOperations<
|
|||
value().set(other.value);
|
||||
}
|
||||
|
||||
void setDataDescriptor(HandleValue v, unsigned attrs) {
|
||||
void setDataDescriptor(JS::HandleValue v, unsigned attrs) {
|
||||
MOZ_ASSERT((attrs & ~(JSPROP_ENUMERATE |
|
||||
JSPROP_PERMANENT |
|
||||
JSPROP_READONLY |
|
||||
|
@ -2810,26 +2815,7 @@ class MutablePropertyDescriptorOperations : public PropertyDescriptorOperations<
|
|||
}
|
||||
};
|
||||
|
||||
} /* namespace JS */
|
||||
|
||||
namespace js {
|
||||
|
||||
template <>
|
||||
class RootedBase<JS::PropertyDescriptor>
|
||||
: public JS::MutablePropertyDescriptorOperations<JS::Rooted<JS::PropertyDescriptor>>
|
||||
{};
|
||||
|
||||
template <>
|
||||
class HandleBase<JS::PropertyDescriptor>
|
||||
: public JS::PropertyDescriptorOperations<JS::Handle<JS::PropertyDescriptor>>
|
||||
{};
|
||||
|
||||
template <>
|
||||
class MutableHandleBase<JS::PropertyDescriptor>
|
||||
: public JS::MutablePropertyDescriptorOperations<JS::MutableHandle<JS::PropertyDescriptor>>
|
||||
{};
|
||||
|
||||
} /* namespace js */
|
||||
} // namespace js
|
||||
|
||||
namespace JS {
|
||||
|
||||
|
|
|
@ -571,7 +571,10 @@ JSCompartment::getNonSyntacticLexicalEnvironment(JSObject* enclosing) const
|
|||
bool
|
||||
JSCompartment::addToVarNames(JSContext* cx, JS::Handle<JSAtom*> name)
|
||||
{
|
||||
if (varNames_.put(name.get()))
|
||||
MOZ_ASSERT(name);
|
||||
MOZ_ASSERT(!isAtomsCompartment());
|
||||
|
||||
if (varNames_.put(name))
|
||||
return true;
|
||||
|
||||
ReportOutOfMemory(cx);
|
||||
|
@ -720,8 +723,9 @@ JSCompartment::sweepAfterMinorGC(JSTracer* trc)
|
|||
{
|
||||
globalWriteBarriered = 0;
|
||||
|
||||
if (innerViews.needsSweepAfterMinorGC())
|
||||
innerViews.sweepAfterMinorGC();
|
||||
InnerViewTable& table = innerViews.get();
|
||||
if (table.needsSweepAfterMinorGC())
|
||||
table.sweepAfterMinorGC();
|
||||
|
||||
crossCompartmentWrappers.sweepAfterMinorGC(trc);
|
||||
}
|
||||
|
@ -803,6 +807,12 @@ JSCompartment::sweepCrossCompartmentWrappers()
|
|||
crossCompartmentWrappers.sweep();
|
||||
}
|
||||
|
||||
void
|
||||
JSCompartment::sweepVarNames()
|
||||
{
|
||||
varNames_.sweep();
|
||||
}
|
||||
|
||||
namespace {
|
||||
struct TraceRootFunctor {
|
||||
JSTracer* trc;
|
||||
|
|
|
@ -654,6 +654,7 @@ struct JSCompartment
|
|||
void sweepDebugEnvironments();
|
||||
void sweepNativeIterators();
|
||||
void sweepTemplateObjects();
|
||||
void sweepVarNames();
|
||||
|
||||
void purge();
|
||||
void clearTables();
|
||||
|
|
|
@ -4870,6 +4870,8 @@ MAKE_GC_SWEEP_TASK(SweepMiscTask);
|
|||
SweepAtomsTask::run()
|
||||
{
|
||||
runtime->sweepAtoms();
|
||||
for (CompartmentsIter comp(runtime, SkipAtoms); !comp.done(); comp.next())
|
||||
comp->sweepVarNames();
|
||||
}
|
||||
|
||||
/* virtual */ void
|
||||
|
|
|
@ -592,21 +592,23 @@ class JSObject : public js::gc::Cell
|
|||
void operator=(const JSObject& other) = delete;
|
||||
};
|
||||
|
||||
template <class U>
|
||||
template <typename Wrapper>
|
||||
template <typename U>
|
||||
MOZ_ALWAYS_INLINE JS::Handle<U*>
|
||||
js::RootedBase<JSObject*>::as() const
|
||||
js::RootedBase<JSObject*, Wrapper>::as() const
|
||||
{
|
||||
const JS::Rooted<JSObject*>& self = *static_cast<const JS::Rooted<JSObject*>*>(this);
|
||||
MOZ_ASSERT(self->is<U>());
|
||||
const Wrapper& self = *static_cast<const Wrapper*>(this);
|
||||
MOZ_ASSERT(self->template is<U>());
|
||||
return Handle<U*>::fromMarkedLocation(reinterpret_cast<U* const*>(self.address()));
|
||||
}
|
||||
|
||||
template <typename Wrapper>
|
||||
template <class U>
|
||||
MOZ_ALWAYS_INLINE JS::Handle<U*>
|
||||
js::HandleBase<JSObject*>::as() const
|
||||
js::HandleBase<JSObject*, Wrapper>::as() const
|
||||
{
|
||||
const JS::Handle<JSObject*>& self = *static_cast<const JS::Handle<JSObject*>*>(this);
|
||||
MOZ_ASSERT(self->is<U>());
|
||||
MOZ_ASSERT(self->template is<U>());
|
||||
return Handle<U*>::fromMarkedLocation(reinterpret_cast<U* const*>(self.address()));
|
||||
}
|
||||
|
||||
|
|
|
@ -350,7 +350,7 @@ ArrayBufferObject::detach(JSContext* cx, Handle<ArrayBufferObject*> buffer,
|
|||
// Update all views of the buffer to account for the buffer having been
|
||||
// detached, and clear the buffer's data and list of views.
|
||||
|
||||
auto& innerViews = cx->compartment()->innerViews;
|
||||
auto& innerViews = cx->compartment()->innerViews.get();
|
||||
if (InnerViewTable::ViewVector* views = innerViews.maybeViewsUnbarriered(buffer)) {
|
||||
for (size_t i = 0; i < views->length(); i++)
|
||||
NoteViewBufferWasDetached((*views)[i], newContents, cx);
|
||||
|
@ -425,7 +425,7 @@ ArrayBufferObject::changeContents(JSContext* cx, BufferContents newContents,
|
|||
setNewData(cx->runtime()->defaultFreeOp(), newContents, ownsState);
|
||||
|
||||
// Update all views.
|
||||
auto& innerViews = cx->compartment()->innerViews;
|
||||
auto& innerViews = cx->compartment()->innerViews.get();
|
||||
if (InnerViewTable::ViewVector* views = innerViews.maybeViewsUnbarriered(this)) {
|
||||
for (size_t i = 0; i < views->length(); i++)
|
||||
changeViewContents(cx, (*views)[i], oldDataPointer, newContents);
|
||||
|
|
|
@ -555,7 +555,6 @@ class InnerViewTable
|
|||
typedef Vector<ArrayBufferViewObject*, 1, SystemAllocPolicy> ViewVector;
|
||||
|
||||
friend class ArrayBufferObject;
|
||||
friend class WeakCacheBase<InnerViewTable>;
|
||||
|
||||
private:
|
||||
struct MapGCPolicy {
|
||||
|
@ -616,23 +615,15 @@ class InnerViewTable
|
|||
size_t sizeOfExcludingThis(mozilla::MallocSizeOf mallocSizeOf);
|
||||
};
|
||||
|
||||
template <>
|
||||
class WeakCacheBase<InnerViewTable>
|
||||
template <typename Wrapper>
|
||||
class MutableWrappedPtrOperations<InnerViewTable, Wrapper>
|
||||
: public WrappedPtrOperations<InnerViewTable, Wrapper>
|
||||
{
|
||||
InnerViewTable& table() {
|
||||
return static_cast<JS::WeakCache<InnerViewTable>*>(this)->get();
|
||||
}
|
||||
const InnerViewTable& table() const {
|
||||
return static_cast<const JS::WeakCache<InnerViewTable>*>(this)->get();
|
||||
return static_cast<Wrapper*>(this)->get();
|
||||
}
|
||||
|
||||
public:
|
||||
InnerViewTable::ViewVector* maybeViewsUnbarriered(ArrayBufferObject* obj) {
|
||||
return table().maybeViewsUnbarriered(obj);
|
||||
}
|
||||
void removeViews(ArrayBufferObject* obj) { table().removeViews(obj); }
|
||||
void sweepAfterMinorGC() { table().sweepAfterMinorGC(); }
|
||||
bool needsSweepAfterMinorGC() const { return table().needsSweepAfterMinorGC(); }
|
||||
size_t sizeOfExcludingThis(mozilla::MallocSizeOf mallocSizeOf) {
|
||||
return table().sizeOfExcludingThis(mallocSizeOf);
|
||||
}
|
||||
|
|
|
@ -1538,11 +1538,7 @@ GetSuperEnvFunction(JSContext* cx, InterpreterRegs& regs)
|
|||
*/
|
||||
|
||||
template<typename T>
|
||||
class ReservedRootedBase {
|
||||
};
|
||||
|
||||
template<typename T>
|
||||
class ReservedRooted : public ReservedRootedBase<T>
|
||||
class ReservedRooted : public RootedBase<T, ReservedRooted<T>>
|
||||
{
|
||||
Rooted<T>* savedRoot;
|
||||
|
||||
|
@ -1570,14 +1566,6 @@ class ReservedRooted : public ReservedRootedBase<T>
|
|||
DECLARE_POINTER_ASSIGN_OPS(ReservedRooted, T)
|
||||
};
|
||||
|
||||
template <>
|
||||
class ReservedRootedBase<Value> : public ValueOperations<ReservedRooted<Value>>
|
||||
{};
|
||||
|
||||
template <>
|
||||
class ReservedRootedBase<Scope*> : public ScopeCastOperation<ReservedRooted<Scope*>>
|
||||
{};
|
||||
|
||||
static MOZ_NEVER_INLINE bool
|
||||
Interpret(JSContext* cx, RunState& state)
|
||||
{
|
||||
|
|
Некоторые файлы не были показаны из-за слишком большого количества измененных файлов Показать больше
Загрузка…
Ссылка в новой задаче