2012-03-09 13:48:50 +04:00
|
|
|
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
|
|
* vim: set ts=8 sw=4 et tw=99 ft=cpp:
|
|
|
|
*
|
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-12-24 12:27:39 +04:00
|
|
|
|
|
|
|
#ifndef js_MemoryMetrics_h
|
|
|
|
#define js_MemoryMetrics_h
|
|
|
|
|
|
|
|
/*
|
|
|
|
* These declarations are not within jsapi.h because they are highly likely
|
|
|
|
* to change in the future. Depend on them at your own risk.
|
|
|
|
*/
|
|
|
|
|
2012-01-11 12:23:08 +04:00
|
|
|
#include <string.h>
|
|
|
|
|
2012-01-11 12:23:08 +04:00
|
|
|
#include "jsalloc.h"
|
2011-12-24 12:27:39 +04:00
|
|
|
#include "jspubtd.h"
|
|
|
|
|
|
|
|
#include "js/Utility.h"
|
2012-01-11 12:23:08 +04:00
|
|
|
#include "js/Vector.h"
|
2011-12-24 12:27:39 +04:00
|
|
|
|
|
|
|
namespace JS {
|
|
|
|
|
|
|
|
/* Data for tracking analysis/inference memory usage. */
|
2012-01-27 07:09:31 +04:00
|
|
|
struct TypeInferenceSizes
|
2011-12-24 12:27:39 +04:00
|
|
|
{
|
2012-01-30 02:13:18 +04:00
|
|
|
size_t scripts;
|
|
|
|
size_t objects;
|
|
|
|
size_t tables;
|
|
|
|
size_t temporary;
|
2012-06-26 04:08:59 +04:00
|
|
|
|
|
|
|
void add(TypeInferenceSizes &sizes) {
|
|
|
|
this->scripts += sizes.scripts;
|
|
|
|
this->objects += sizes.objects;
|
|
|
|
this->tables += sizes.tables;
|
|
|
|
this->temporary += sizes.temporary;
|
|
|
|
}
|
2011-12-24 12:27:39 +04:00
|
|
|
};
|
|
|
|
|
2012-05-16 06:29:14 +04:00
|
|
|
// These measurements relate directly to the JSRuntime, and not to
|
|
|
|
// compartments within it.
|
|
|
|
struct RuntimeSizes
|
|
|
|
{
|
|
|
|
RuntimeSizes()
|
|
|
|
: object(0)
|
|
|
|
, atomsTable(0)
|
|
|
|
, contexts(0)
|
|
|
|
, dtoa(0)
|
|
|
|
, temporary(0)
|
|
|
|
, mjitCode(0)
|
|
|
|
, regexpCode(0)
|
|
|
|
, unusedCodeMemory(0)
|
|
|
|
, stackCommitted(0)
|
|
|
|
, gcMarker(0)
|
2012-05-16 06:30:28 +04:00
|
|
|
, mathCache(0)
|
|
|
|
, scriptFilenames(0)
|
2012-07-20 22:18:08 +04:00
|
|
|
, scriptSources(0)
|
2012-05-16 06:30:28 +04:00
|
|
|
, compartmentObjects(0)
|
2012-05-16 06:29:14 +04:00
|
|
|
{}
|
|
|
|
|
|
|
|
size_t object;
|
|
|
|
size_t atomsTable;
|
|
|
|
size_t contexts;
|
|
|
|
size_t dtoa;
|
|
|
|
size_t temporary;
|
|
|
|
size_t mjitCode;
|
|
|
|
size_t regexpCode;
|
|
|
|
size_t unusedCodeMemory;
|
|
|
|
size_t stackCommitted;
|
|
|
|
size_t gcMarker;
|
2012-05-16 06:30:28 +04:00
|
|
|
size_t mathCache;
|
|
|
|
size_t scriptFilenames;
|
2012-07-20 22:18:08 +04:00
|
|
|
size_t scriptSources;
|
2012-05-16 06:30:28 +04:00
|
|
|
|
|
|
|
// This is the exception to the "RuntimeSizes doesn't measure things within
|
|
|
|
// compartments" rule. We combine the sizes of all the JSCompartment
|
|
|
|
// objects into a single measurement because each one is fairly small, and
|
|
|
|
// they're all the same size.
|
|
|
|
size_t compartmentObjects;
|
2012-05-16 06:29:14 +04:00
|
|
|
};
|
|
|
|
|
2012-01-11 12:23:08 +04:00
|
|
|
struct CompartmentStats
|
|
|
|
{
|
2012-03-09 13:48:50 +04:00
|
|
|
CompartmentStats() {
|
2012-01-11 12:23:08 +04:00
|
|
|
memset(this, 0, sizeof(*this));
|
|
|
|
}
|
|
|
|
|
2012-07-06 08:12:37 +04:00
|
|
|
// These fields can be used by embedders.
|
|
|
|
void *extra1;
|
|
|
|
void *extra2;
|
2012-06-26 04:08:59 +04:00
|
|
|
|
|
|
|
// If you add a new number, remember to update add() and maybe
|
|
|
|
// gcHeapThingsSize()!
|
2012-06-26 04:06:50 +04:00
|
|
|
size_t gcHeapArenaAdmin;
|
2012-06-26 04:06:50 +04:00
|
|
|
size_t gcHeapUnusedGcThings;
|
2012-01-30 02:13:18 +04:00
|
|
|
|
|
|
|
size_t gcHeapObjectsNonFunction;
|
|
|
|
size_t gcHeapObjectsFunction;
|
|
|
|
size_t gcHeapStrings;
|
|
|
|
size_t gcHeapShapesTree;
|
|
|
|
size_t gcHeapShapesDict;
|
|
|
|
size_t gcHeapShapesBase;
|
|
|
|
size_t gcHeapScripts;
|
|
|
|
size_t gcHeapTypeObjects;
|
2012-05-18 01:24:32 +04:00
|
|
|
#if JS_HAS_XML_SUPPORT
|
2012-01-30 02:13:18 +04:00
|
|
|
size_t gcHeapXML;
|
2012-05-18 01:24:32 +04:00
|
|
|
#endif
|
2012-01-30 02:13:18 +04:00
|
|
|
|
|
|
|
size_t objectSlots;
|
|
|
|
size_t objectElements;
|
2012-01-31 06:12:03 +04:00
|
|
|
size_t objectMisc;
|
2012-04-20 08:01:33 +04:00
|
|
|
size_t objectPrivate;
|
2012-01-30 02:13:18 +04:00
|
|
|
size_t stringChars;
|
|
|
|
size_t shapesExtraTreeTables;
|
|
|
|
size_t shapesExtraDictTables;
|
|
|
|
size_t shapesExtraTreeShapeKids;
|
|
|
|
size_t shapesCompartmentTables;
|
|
|
|
size_t scriptData;
|
|
|
|
size_t mjitData;
|
2012-05-16 06:31:01 +04:00
|
|
|
size_t crossCompartmentWrappers;
|
2012-05-03 11:12:47 +04:00
|
|
|
|
2012-01-27 07:09:31 +04:00
|
|
|
TypeInferenceSizes typeInferenceSizes;
|
2012-06-26 04:08:59 +04:00
|
|
|
|
|
|
|
// Add cStats's numbers to this object's numbers.
|
|
|
|
void add(CompartmentStats &cStats) {
|
|
|
|
#define ADD(x) this->x += cStats.x
|
|
|
|
|
|
|
|
ADD(gcHeapArenaAdmin);
|
|
|
|
ADD(gcHeapUnusedGcThings);
|
|
|
|
|
|
|
|
ADD(gcHeapObjectsNonFunction);
|
|
|
|
ADD(gcHeapObjectsFunction);
|
|
|
|
ADD(gcHeapStrings);
|
|
|
|
ADD(gcHeapShapesTree);
|
|
|
|
ADD(gcHeapShapesDict);
|
|
|
|
ADD(gcHeapShapesBase);
|
|
|
|
ADD(gcHeapScripts);
|
|
|
|
ADD(gcHeapTypeObjects);
|
|
|
|
#if JS_HAS_XML_SUPPORT
|
|
|
|
ADD(gcHeapXML);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
ADD(objectSlots);
|
|
|
|
ADD(objectElements);
|
|
|
|
ADD(objectMisc);
|
2012-04-20 08:01:33 +04:00
|
|
|
ADD(objectPrivate);
|
2012-06-26 04:08:59 +04:00
|
|
|
ADD(stringChars);
|
|
|
|
ADD(shapesExtraTreeTables);
|
|
|
|
ADD(shapesExtraDictTables);
|
|
|
|
ADD(shapesExtraTreeShapeKids);
|
|
|
|
ADD(shapesCompartmentTables);
|
|
|
|
ADD(scriptData);
|
|
|
|
ADD(mjitData);
|
|
|
|
ADD(crossCompartmentWrappers);
|
|
|
|
|
|
|
|
#undef ADD
|
|
|
|
|
|
|
|
typeInferenceSizes.add(cStats.typeInferenceSizes);
|
|
|
|
}
|
|
|
|
|
|
|
|
// The size of all the live things in the GC heap.
|
|
|
|
size_t gcHeapThingsSize();
|
2012-01-11 12:23:08 +04:00
|
|
|
};
|
|
|
|
|
2012-01-30 02:11:32 +04:00
|
|
|
struct RuntimeStats
|
2012-01-11 12:23:08 +04:00
|
|
|
{
|
2012-03-09 13:48:50 +04:00
|
|
|
RuntimeStats(JSMallocSizeOfFun mallocSizeOf)
|
2012-05-16 06:29:14 +04:00
|
|
|
: runtime()
|
2012-01-11 12:23:08 +04:00
|
|
|
, gcHeapChunkTotal(0)
|
2012-06-26 04:09:00 +04:00
|
|
|
, gcHeapDecommittedArenas(0)
|
2012-06-26 04:08:59 +04:00
|
|
|
, gcHeapUnusedChunks(0)
|
|
|
|
, gcHeapUnusedArenas(0)
|
2012-06-26 04:06:50 +04:00
|
|
|
, gcHeapUnusedGcThings(0)
|
2012-01-11 12:23:08 +04:00
|
|
|
, gcHeapChunkAdmin(0)
|
2012-06-26 04:08:59 +04:00
|
|
|
, gcHeapGcThings(0)
|
|
|
|
, totals()
|
2012-01-11 12:23:08 +04:00
|
|
|
, compartmentStatsVector()
|
|
|
|
, currCompartmentStats(NULL)
|
|
|
|
, mallocSizeOf(mallocSizeOf)
|
|
|
|
{}
|
|
|
|
|
2012-05-18 04:08:57 +04:00
|
|
|
RuntimeSizes runtime;
|
2012-05-16 06:29:14 +04:00
|
|
|
|
2012-06-26 04:08:59 +04:00
|
|
|
// If you add a new number, remember to update the constructor!
|
|
|
|
|
|
|
|
// Here's a useful breakdown of the GC heap.
|
|
|
|
//
|
|
|
|
// - rtStats.gcHeapChunkTotal
|
|
|
|
// - decommitted bytes
|
2012-06-26 04:09:00 +04:00
|
|
|
// - rtStats.gcHeapDecommittedArenas (decommitted arenas in non-empty chunks)
|
2012-06-26 04:08:59 +04:00
|
|
|
// - unused bytes
|
|
|
|
// - rtStats.gcHeapUnusedChunks (empty chunks)
|
|
|
|
// - rtStats.gcHeapUnusedArenas (empty arenas within non-empty chunks)
|
|
|
|
// - rtStats.total.gcHeapUnusedGcThings (empty GC thing slots within non-empty arenas)
|
|
|
|
// - used bytes
|
|
|
|
// - rtStats.gcHeapChunkAdmin
|
|
|
|
// - rtStats.total.gcHeapArenaAdmin
|
|
|
|
// - rtStats.gcHeapGcThings (in-use GC things)
|
2012-06-26 04:09:00 +04:00
|
|
|
//
|
|
|
|
// It's possible that some arenas in empty chunks may be decommitted, but
|
|
|
|
// we don't count those under rtStats.gcHeapDecommittedArenas because (a)
|
|
|
|
// it's rare, and (b) this means that rtStats.gcHeapUnusedChunks is a
|
|
|
|
// multiple of the chunk size, which is good.
|
2012-06-26 04:08:59 +04:00
|
|
|
|
2012-01-30 02:13:18 +04:00
|
|
|
size_t gcHeapChunkTotal;
|
2012-06-26 04:09:00 +04:00
|
|
|
size_t gcHeapDecommittedArenas;
|
2012-06-26 04:08:59 +04:00
|
|
|
size_t gcHeapUnusedChunks;
|
|
|
|
size_t gcHeapUnusedArenas;
|
2012-06-26 04:06:50 +04:00
|
|
|
size_t gcHeapUnusedGcThings;
|
2012-01-30 02:13:18 +04:00
|
|
|
size_t gcHeapChunkAdmin;
|
2012-06-26 04:08:59 +04:00
|
|
|
size_t gcHeapGcThings;
|
2012-01-11 12:23:08 +04:00
|
|
|
|
2012-06-26 04:08:59 +04:00
|
|
|
// The sum of all compartment's measurements.
|
|
|
|
CompartmentStats totals;
|
|
|
|
|
2012-01-11 12:23:08 +04:00
|
|
|
js::Vector<CompartmentStats, 0, js::SystemAllocPolicy> compartmentStatsVector;
|
|
|
|
CompartmentStats *currCompartmentStats;
|
|
|
|
|
|
|
|
JSMallocSizeOfFun mallocSizeOf;
|
2012-03-09 13:48:50 +04:00
|
|
|
|
|
|
|
virtual void initExtraCompartmentStats(JSCompartment *c, CompartmentStats *cstats) = 0;
|
2012-01-11 12:23:08 +04:00
|
|
|
};
|
|
|
|
|
2012-01-11 20:21:26 +04:00
|
|
|
#ifdef JS_THREADSAFE
|
|
|
|
|
2012-04-20 08:01:33 +04:00
|
|
|
class ObjectPrivateVisitor
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
// Within CollectRuntimeStats, this method is called for each JS object
|
|
|
|
// that has a private slot containing an nsISupports pointer.
|
|
|
|
virtual size_t sizeOfIncludingThis(void *aSupports) = 0;
|
|
|
|
};
|
|
|
|
|
2012-01-11 12:23:08 +04:00
|
|
|
extern JS_PUBLIC_API(bool)
|
2012-04-20 08:01:33 +04:00
|
|
|
CollectRuntimeStats(JSRuntime *rt, RuntimeStats *rtStats, ObjectPrivateVisitor *opv);
|
2012-01-11 12:23:08 +04:00
|
|
|
|
2012-03-01 00:23:53 +04:00
|
|
|
extern JS_PUBLIC_API(int64_t)
|
|
|
|
GetExplicitNonHeapForRuntime(JSRuntime *rt, JSMallocSizeOfFun mallocSizeOf);
|
2012-01-11 12:23:08 +04:00
|
|
|
|
2012-01-11 20:21:26 +04:00
|
|
|
#endif /* JS_THREADSAFE */
|
|
|
|
|
2011-12-24 12:27:39 +04:00
|
|
|
extern JS_PUBLIC_API(size_t)
|
|
|
|
SystemCompartmentCount(const JSRuntime *rt);
|
|
|
|
|
|
|
|
extern JS_PUBLIC_API(size_t)
|
|
|
|
UserCompartmentCount(const JSRuntime *rt);
|
|
|
|
|
|
|
|
} // namespace JS
|
|
|
|
|
|
|
|
#endif // js_MemoryMetrics_h
|