2015-05-03 22:32:37 +03:00
|
|
|
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
|
|
|
|
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
|
2012-05-21 15:12:37 +04:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
2011-05-26 23:58:35 +04:00
|
|
|
|
|
|
|
#ifndef nsWrapperCacheInline_h___
|
|
|
|
#define nsWrapperCacheInline_h___
|
|
|
|
|
|
|
|
#include "nsWrapperCache.h"
|
2014-04-12 02:19:05 +04:00
|
|
|
#include "js/TracingAPI.h"
|
2011-05-26 23:58:35 +04:00
|
|
|
|
2017-04-26 13:18:39 +03:00
|
|
|
inline JSObject*
|
|
|
|
nsWrapperCache::GetWrapperPreserveColor() const
|
|
|
|
{
|
|
|
|
JSObject* obj = mWrapper;
|
|
|
|
if (obj && js::gc::EdgeNeedsSweepUnbarriered(&obj)) {
|
|
|
|
// The object has been found to be dead and is in the process of being
|
|
|
|
// finalized, so don't let the caller see it. As an optimisation, remove it
|
|
|
|
// from the cache so we don't have to do this check in future.
|
|
|
|
const_cast<nsWrapperCache*>(this)->ClearWrapper();
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
MOZ_ASSERT(obj == mWrapper);
|
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
2011-05-26 23:58:35 +04:00
|
|
|
inline JSObject*
|
|
|
|
nsWrapperCache::GetWrapper() const
|
|
|
|
{
|
|
|
|
JSObject* obj = GetWrapperPreserveColor();
|
2013-09-09 07:28:48 +04:00
|
|
|
if (obj) {
|
|
|
|
JS::ExposeObjectToActiveJS(obj);
|
|
|
|
}
|
2011-05-26 23:58:35 +04:00
|
|
|
return obj;
|
|
|
|
}
|
|
|
|
|
2011-12-30 01:21:33 +04:00
|
|
|
inline bool
|
2017-01-25 04:38:58 +03:00
|
|
|
nsWrapperCache::HasKnownLiveWrapper() const
|
2011-12-30 01:21:33 +04:00
|
|
|
{
|
2017-01-25 04:38:58 +03:00
|
|
|
// If we have a wrapper and it's not gray in the GC-marking sense, that means
|
|
|
|
// that we can't be cycle-collected. That's because the wrapper is being kept
|
|
|
|
// alive by the JS engine (and not just due to being traced from some
|
|
|
|
// cycle-collectable thing), and the wrapper holds us alive, so we know we're
|
|
|
|
// not collectable.
|
2011-12-30 01:21:33 +04:00
|
|
|
JSObject* o = GetWrapperPreserveColor();
|
2014-12-05 20:38:34 +03:00
|
|
|
return o && !JS::ObjectIsMarkedGray(o);
|
2011-12-30 01:21:33 +04:00
|
|
|
}
|
|
|
|
|
2012-09-29 02:29:37 +04:00
|
|
|
static void
|
2014-12-05 20:38:34 +03:00
|
|
|
SearchGray(JS::GCCellPtr aGCThing, const char* aName, void* aClosure)
|
2012-09-29 02:29:37 +04:00
|
|
|
{
|
|
|
|
bool* hasGrayObjects = static_cast<bool*>(aClosure);
|
2014-12-05 20:38:34 +03:00
|
|
|
if (!*hasGrayObjects && aGCThing && JS::GCThingIsMarkedGray(aGCThing)) {
|
2012-09-29 02:29:37 +04:00
|
|
|
*hasGrayObjects = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-24 23:35:34 +04:00
|
|
|
inline bool
|
|
|
|
nsWrapperCache::HasNothingToTrace(nsISupports* aThis)
|
|
|
|
{
|
|
|
|
nsXPCOMCycleCollectionParticipant* participant = nullptr;
|
|
|
|
CallQueryInterface(aThis, &participant);
|
|
|
|
bool hasGrayObjects = false;
|
|
|
|
participant->Trace(aThis, TraceCallbackFunc(SearchGray), &hasGrayObjects);
|
|
|
|
return !hasGrayObjects;
|
|
|
|
}
|
|
|
|
|
2012-09-29 02:29:37 +04:00
|
|
|
inline bool
|
2017-01-25 04:39:37 +03:00
|
|
|
nsWrapperCache::HasKnownLiveWrapperAndDoesNotNeedTracing(nsISupports* aThis)
|
2012-09-29 02:29:37 +04:00
|
|
|
{
|
2017-01-25 04:38:58 +03:00
|
|
|
return HasKnownLiveWrapper() && HasNothingToTrace(aThis);
|
2012-09-29 02:29:37 +04:00
|
|
|
}
|
|
|
|
|
2017-01-25 04:33:54 +03:00
|
|
|
inline void
|
|
|
|
nsWrapperCache::MarkWrapperLive()
|
|
|
|
{
|
|
|
|
// Just call GetWrapper and ignore the return value. It will do the
|
|
|
|
// gray-unmarking for us.
|
|
|
|
GetWrapper();
|
|
|
|
}
|
|
|
|
|
2011-05-26 23:58:35 +04:00
|
|
|
#endif /* nsWrapperCache_h___ */
|