#include "gc/WeakMap-inl.h"
#include <string.h>
#include "jsapi.h"
#include "jsfriendapi.h"
#include "gc/PublicIterators.h"
#include "js/Wrapper.h"
#include "vm/GlobalObject.h"
#include "vm/JSContext.h"
#include "vm/JSObject.h"
#include "vm/JSObject-inl.h"
using namespace js;
using namespace js::gc;
WeakMapBase::WeakMapBase(JSObject* memOf, Zone* zone)
: memberOf(memOf), zone_(zone), marked(false), markColor(MarkColor::Black) {
MOZ_ASSERT_IF(memberOf, memberOf->compartment()->zone() == zone);
}
WeakMapBase::~WeakMapBase() {
MOZ_ASSERT(CurrentThreadIsGCSweeping() || CurrentThreadCanAccessZone(zone_));
}
void WeakMapBase::unmarkZone(JS::Zone* zone) {
for (WeakMapBase* m : zone->gcWeakMapList()) {
m->marked = false;
}
}
void WeakMapBase::traceZone(JS::Zone* zone, JSTracer* tracer) {
MOZ_ASSERT(tracer->weakMapAction() != DoNotTraceWeakMaps);
for (WeakMapBase* m : zone->gcWeakMapList()) {
m->trace(tracer);
TraceNullableEdge(tracer, &m->memberOf, "memberOf");
}
}
#if defined(JS_GC_ZEAL) || defined(DEBUG)
bool WeakMapBase::checkMarkingForZone(JS::Zone* zone) {
MOZ_ASSERT(zone->isGCMarking());
bool ok = true;
for (WeakMapBase* m : zone->gcWeakMapList()) {
if (m->marked && !m->checkMarking()) {
ok = false;
}
}
return ok;
}
#endif
bool WeakMapBase::markZoneIteratively(JS::Zone* zone, GCMarker* marker) {
bool markedAny = false;
for (WeakMapBase* m : zone->gcWeakMapList()) {
if (m->marked && m->markIteratively(marker)) {
markedAny = true;
}
}
return markedAny;
}
bool WeakMapBase::findSweepGroupEdges(JS::Zone* zone) {
for (WeakMapBase* m : zone->gcWeakMapList()) {
if (!m->findZoneEdges()) {
return false;
}
}
return true;
}
void WeakMapBase::sweepZone(JS::Zone* zone) {
for (WeakMapBase* m = zone->gcWeakMapList().getFirst(); m;) {
WeakMapBase* next = m->getNext();
if (m->marked) {
m->sweep();
} else {
m->clearAndCompact();
m->removeFrom(zone->gcWeakMapList());
}
m = next;
}
#ifdef DEBUG
for (WeakMapBase* m : zone->gcWeakMapList()) {
MOZ_ASSERT(m->isInList() && m->marked);
}
#endif
}
void WeakMapBase::traceAllMappings(WeakMapTracer* tracer) {
JSRuntime* rt = tracer->runtime;
for (ZonesIter zone(rt, SkipAtoms); !zone.done(); zone.next()) {
for (WeakMapBase* m : zone->gcWeakMapList()) {
JS::AutoSuppressGCAnalysis nogc;
m->traceMappings(tracer);
}
}
}
bool WeakMapBase::saveZoneMarkedWeakMaps(JS::Zone* zone,
WeakMapSet& markedWeakMaps) {
for (WeakMapBase* m : zone->gcWeakMapList()) {
if (m->marked && !markedWeakMaps.put(m)) {
return false;
}
}
return true;
}
void WeakMapBase::restoreMarkedWeakMaps(WeakMapSet& markedWeakMaps) {
for (WeakMapSet::Range r = markedWeakMaps.all(); !r.empty(); r.popFront()) {
WeakMapBase* map = r.front();
MOZ_ASSERT(map->zone()->isGCMarking());
MOZ_ASSERT(!map->marked);
map->marked = true;
}
}
bool ObjectValueMap::findZoneEdges() {
JS::AutoSuppressGCAnalysis nogc;
for (Range r = all(); !r.empty(); r.popFront()) {
JSObject* key = r.front().key();
if (key->asTenured().isMarkedBlack()) {
continue;
}
JSObject* delegate = getDelegate(key);
if (!delegate) {
continue;
}
Zone* delegateZone = delegate->zone();
if (delegateZone == zone() || !delegateZone->isGCMarking()) {
continue;
}
if (!delegateZone->addSweepGroupEdgeTo(key->zone())) {
return false;
}
}
return true;
}
ObjectWeakMap::ObjectWeakMap(JSContext* cx) : map(cx, nullptr) {}
JSObject* ObjectWeakMap::lookup(const JSObject* obj) {
if (ObjectValueMap::Ptr p = map.lookup(const_cast<JSObject*>(obj))) {
return &p->value().toObject();
}
return nullptr;
}
bool ObjectWeakMap::add(JSContext* cx, JSObject* obj, JSObject* target) {
MOZ_ASSERT(obj && target);
MOZ_ASSERT(!map.has(obj));
if (!map.put(obj, ObjectValue(*target))) {
ReportOutOfMemory(cx);
return false;
}
return true;
}
void ObjectWeakMap::clear() { map.clear(); }
void ObjectWeakMap::trace(JSTracer* trc) { map.trace(trc); }
size_t ObjectWeakMap::sizeOfExcludingThis(mozilla::MallocSizeOf mallocSizeOf) {
return map.shallowSizeOfExcludingThis(mallocSizeOf);
}
#ifdef JSGC_HASH_TABLE_CHECKS
void ObjectWeakMap::checkAfterMovingGC() {
for (ObjectValueMap::Range r = map.all(); !r.empty(); r.popFront()) {
CheckGCThingAfterMovingGC(r.front().key().get());
CheckGCThingAfterMovingGC(&r.front().value().toObject());
}
}
#endif