2015-09-26 06:09:58 +03: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/. */
|
|
|
|
"use strict";
|
|
|
|
|
2015-10-27 23:24:55 +03:00
|
|
|
const { assert, reportException } = require("devtools/shared/DevToolsUtils");
|
2015-11-13 22:20:45 +03:00
|
|
|
const { censusIsUpToDate, getSnapshot, breakdownEquals, createSnapshot } = require("../utils");
|
2015-10-15 09:13:17 +03:00
|
|
|
const { actions, snapshotState: states } = require("../constants");
|
2015-11-13 22:20:45 +03:00
|
|
|
const { toggleDiffing } = require("./diffing");
|
2015-10-15 09:13:17 +03:00
|
|
|
|
|
|
|
/**
|
2015-11-13 22:20:45 +03:00
|
|
|
* A series of actions are fired from this task to save, read and generate the
|
|
|
|
* initial census from a snapshot.
|
2015-10-15 09:13:17 +03:00
|
|
|
*
|
|
|
|
* @param {MemoryFront}
|
|
|
|
* @param {HeapAnalysesClient}
|
|
|
|
* @param {Object}
|
|
|
|
*/
|
2015-10-17 05:15:54 +03:00
|
|
|
const takeSnapshotAndCensus = exports.takeSnapshotAndCensus = function (front, heapWorker) {
|
|
|
|
return function *(dispatch, getState) {
|
2015-11-13 22:20:45 +03:00
|
|
|
const id = yield dispatch(takeSnapshot(front));
|
|
|
|
if (id === null) {
|
|
|
|
return;
|
|
|
|
}
|
2015-10-27 23:24:55 +03:00
|
|
|
|
2015-11-13 22:20:45 +03:00
|
|
|
yield dispatch(readSnapshot(heapWorker, id));
|
|
|
|
if (getSnapshot(getState(), id).state === states.READ) {
|
|
|
|
yield dispatch(takeCensus(heapWorker, id));
|
2015-10-27 23:24:55 +03:00
|
|
|
}
|
2015-10-15 09:13:17 +03:00
|
|
|
};
|
|
|
|
};
|
2015-09-26 06:09:58 +03:00
|
|
|
|
2015-10-17 05:15:54 +03:00
|
|
|
/**
|
|
|
|
* Selects a snapshot and if the snapshot's census is using a different
|
|
|
|
* breakdown, take a new census.
|
|
|
|
*
|
2015-11-13 22:20:45 +03:00
|
|
|
* @param {HeapAnalysesClient} heapWorker
|
|
|
|
* @param {snapshotId} id
|
2015-10-17 05:15:54 +03:00
|
|
|
*/
|
2015-11-13 22:20:45 +03:00
|
|
|
const selectSnapshotAndRefresh = exports.selectSnapshotAndRefresh = function (heapWorker, id) {
|
2015-10-17 05:15:54 +03:00
|
|
|
return function *(dispatch, getState) {
|
2015-11-13 22:20:45 +03:00
|
|
|
if (getState().diffing) {
|
|
|
|
dispatch(toggleDiffing());
|
|
|
|
}
|
|
|
|
|
|
|
|
dispatch(selectSnapshot(id));
|
2015-10-27 23:24:55 +03:00
|
|
|
yield dispatch(refreshSelectedCensus(heapWorker));
|
2015-10-17 05:15:54 +03:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2015-10-10 18:09:28 +03:00
|
|
|
/**
|
2015-11-13 22:20:45 +03:00
|
|
|
* Take a snapshot and return its id on success, or null on failure.
|
|
|
|
*
|
|
|
|
* @param {MemoryFront} front
|
|
|
|
* @returns {Number|null}
|
2015-10-10 18:09:28 +03:00
|
|
|
*/
|
2015-10-17 05:15:54 +03:00
|
|
|
const takeSnapshot = exports.takeSnapshot = function (front) {
|
|
|
|
return function *(dispatch, getState) {
|
2015-11-13 22:20:45 +03:00
|
|
|
if (getState().diffing) {
|
|
|
|
dispatch(toggleDiffing());
|
|
|
|
}
|
|
|
|
|
|
|
|
const snapshot = createSnapshot();
|
|
|
|
const id = snapshot.id;
|
2015-10-15 09:13:17 +03:00
|
|
|
dispatch({ type: actions.TAKE_SNAPSHOT_START, snapshot });
|
2015-11-13 22:20:45 +03:00
|
|
|
dispatch(selectSnapshot(id));
|
2015-10-15 09:13:17 +03:00
|
|
|
|
2015-10-27 23:24:55 +03:00
|
|
|
let path;
|
|
|
|
try {
|
|
|
|
path = yield front.saveHeapSnapshot();
|
|
|
|
} catch (error) {
|
|
|
|
reportException("takeSnapshot", error);
|
2015-11-13 22:20:45 +03:00
|
|
|
dispatch({ type: actions.SNAPSHOT_ERROR, id, error });
|
|
|
|
return null;
|
2015-10-27 23:24:55 +03:00
|
|
|
}
|
2015-10-15 09:13:17 +03:00
|
|
|
|
2015-11-13 22:20:45 +03:00
|
|
|
dispatch({ type: actions.TAKE_SNAPSHOT_END, id, path });
|
|
|
|
return snapshot.id;
|
2015-10-15 09:13:17 +03:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Reads a snapshot into memory; necessary to do before taking
|
|
|
|
* a census on the snapshot. May only be called once per snapshot.
|
|
|
|
*
|
2015-11-13 22:20:45 +03:00
|
|
|
* @param {HeapAnalysesClient} heapWorker
|
|
|
|
* @param {snapshotId} id
|
2015-10-15 09:13:17 +03:00
|
|
|
*/
|
2015-11-13 22:20:45 +03:00
|
|
|
const readSnapshot = exports.readSnapshot = function readSnapshot (heapWorker, id) {
|
2015-10-17 05:15:54 +03:00
|
|
|
return function *(dispatch, getState) {
|
2015-11-13 22:20:45 +03:00
|
|
|
const snapshot = getSnapshot(getState(), id);
|
2015-11-11 20:52:36 +03:00
|
|
|
assert([states.SAVED, states.IMPORTING].includes(snapshot.state),
|
2015-10-27 23:24:55 +03:00
|
|
|
`Should only read a snapshot once. Found snapshot in state ${snapshot.state}`);
|
2015-10-15 09:13:17 +03:00
|
|
|
|
2015-10-29 07:27:46 +03:00
|
|
|
let creationTime;
|
|
|
|
|
2015-11-13 22:20:45 +03:00
|
|
|
dispatch({ type: actions.READ_SNAPSHOT_START, id });
|
2015-10-27 23:24:55 +03:00
|
|
|
try {
|
|
|
|
yield heapWorker.readHeapSnapshot(snapshot.path);
|
2015-10-29 07:27:46 +03:00
|
|
|
creationTime = yield heapWorker.getCreationTime(snapshot.path);
|
2015-10-27 23:24:55 +03:00
|
|
|
} catch (error) {
|
|
|
|
reportException("readSnapshot", error);
|
2015-11-13 22:20:45 +03:00
|
|
|
dispatch({ type: actions.SNAPSHOT_ERROR, id, error });
|
2015-10-27 23:24:55 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-11-13 22:20:45 +03:00
|
|
|
dispatch({ type: actions.READ_SNAPSHOT_END, id, creationTime });
|
2015-10-15 09:13:17 +03:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {HeapAnalysesClient} heapWorker
|
2015-11-13 22:20:45 +03:00
|
|
|
* @param {snapshotId} id
|
2015-10-15 09:13:17 +03:00
|
|
|
*
|
2015-10-17 05:15:54 +03:00
|
|
|
* @see {Snapshot} model defined in devtools/client/memory/models.js
|
2015-10-15 09:13:17 +03:00
|
|
|
* @see `devtools/shared/heapsnapshot/HeapAnalysesClient.js`
|
|
|
|
* @see `js/src/doc/Debugger/Debugger.Memory.md` for breakdown details
|
|
|
|
*/
|
2015-11-13 22:20:45 +03:00
|
|
|
const takeCensus = exports.takeCensus = function (heapWorker, id) {
|
2015-10-17 05:15:54 +03:00
|
|
|
return function *(dispatch, getState) {
|
2015-11-13 22:20:45 +03:00
|
|
|
const snapshot = getSnapshot(getState(), id);
|
2015-10-15 09:13:17 +03:00
|
|
|
assert([states.READ, states.SAVED_CENSUS].includes(snapshot.state),
|
2015-10-27 23:24:55 +03:00
|
|
|
`Can only take census of snapshots in READ or SAVED_CENSUS state, found ${snapshot.state}`);
|
2015-10-15 09:13:17 +03:00
|
|
|
|
2015-11-13 22:20:45 +03:00
|
|
|
let report;
|
2015-10-27 19:11:05 +03:00
|
|
|
let inverted = getState().inverted;
|
2015-10-17 05:15:54 +03:00
|
|
|
let breakdown = getState().breakdown;
|
2015-11-05 03:12:31 +03:00
|
|
|
let filter = getState().filter;
|
2015-10-17 05:15:54 +03:00
|
|
|
|
2015-11-05 03:12:31 +03:00
|
|
|
// If breakdown, filter and inversion haven't changed, don't do anything.
|
2015-11-13 22:20:45 +03:00
|
|
|
if (censusIsUpToDate(inverted, filter, breakdown, snapshot.census)) {
|
2015-10-17 05:15:54 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Keep taking a census if the breakdown changes during. Recheck
|
|
|
|
// that the breakdown used for the census is the same as
|
|
|
|
// the state's breakdown.
|
|
|
|
do {
|
2015-10-27 19:11:05 +03:00
|
|
|
inverted = getState().inverted;
|
2015-10-17 05:15:54 +03:00
|
|
|
breakdown = getState().breakdown;
|
2015-11-05 03:12:31 +03:00
|
|
|
filter = getState().filter;
|
|
|
|
|
|
|
|
dispatch({
|
|
|
|
type: actions.TAKE_CENSUS_START,
|
2015-11-13 22:20:45 +03:00
|
|
|
id,
|
2015-11-05 03:12:31 +03:00
|
|
|
inverted,
|
|
|
|
filter,
|
|
|
|
breakdown
|
|
|
|
});
|
|
|
|
|
2015-10-27 19:11:05 +03:00
|
|
|
let opts = inverted ? { asInvertedTreeNode: true } : { asTreeNode: true };
|
2015-11-05 03:12:31 +03:00
|
|
|
opts.filter = filter || null;
|
2015-10-27 23:24:55 +03:00
|
|
|
|
|
|
|
try {
|
2015-11-13 22:20:45 +03:00
|
|
|
report = yield heapWorker.takeCensus(snapshot.path, { breakdown }, opts);
|
|
|
|
} catch (error) {
|
2015-10-27 23:24:55 +03:00
|
|
|
reportException("takeCensus", error);
|
2015-11-13 22:20:45 +03:00
|
|
|
dispatch({ type: actions.SNAPSHOT_ERROR, id, error });
|
2015-10-27 23:24:55 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (inverted !== getState().inverted ||
|
2015-11-05 03:12:31 +03:00
|
|
|
filter !== getState().filter ||
|
2015-10-27 23:24:55 +03:00
|
|
|
!breakdownEquals(breakdown, getState().breakdown));
|
2015-10-15 09:13:17 +03:00
|
|
|
|
2015-11-05 03:12:31 +03:00
|
|
|
dispatch({
|
|
|
|
type: actions.TAKE_CENSUS_END,
|
2015-11-13 22:20:45 +03:00
|
|
|
id,
|
2015-11-05 03:12:31 +03:00
|
|
|
breakdown,
|
|
|
|
inverted,
|
|
|
|
filter,
|
2015-11-13 22:20:45 +03:00
|
|
|
report
|
2015-11-05 03:12:31 +03:00
|
|
|
});
|
2015-10-27 19:11:05 +03:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Refresh the selected snapshot's census data, if need be (for example,
|
|
|
|
* breakdown configuration changed).
|
|
|
|
*
|
|
|
|
* @param {HeapAnalysesClient} heapWorker
|
|
|
|
*/
|
|
|
|
const refreshSelectedCensus = exports.refreshSelectedCensus = function (heapWorker) {
|
|
|
|
return function*(dispatch, getState) {
|
|
|
|
let snapshot = getState().snapshots.find(s => s.selected);
|
|
|
|
|
|
|
|
// Intermediate snapshot states will get handled by the task action that is
|
|
|
|
// orchestrating them. For example, if the snapshot's state is
|
|
|
|
// SAVING_CENSUS, then the takeCensus action will keep taking a census until
|
|
|
|
// the inverted property matches the inverted state. If the snapshot is
|
|
|
|
// still in the process of being saved or read, the takeSnapshotAndCensus
|
|
|
|
// task action will follow through and ensure that a census is taken.
|
|
|
|
if (snapshot && snapshot.state === states.SAVED_CENSUS) {
|
2015-11-13 22:20:45 +03:00
|
|
|
yield dispatch(takeCensus(heapWorker, snapshot.id));
|
2015-10-27 19:11:05 +03:00
|
|
|
}
|
2015-09-26 06:09:58 +03:00
|
|
|
};
|
|
|
|
};
|
2015-10-10 18:09:28 +03:00
|
|
|
|
|
|
|
/**
|
2015-11-13 22:20:45 +03:00
|
|
|
* Select the snapshot with the given id.
|
|
|
|
*
|
|
|
|
* @param {snapshotId} id
|
2015-10-17 05:15:54 +03:00
|
|
|
* @see {Snapshot} model defined in devtools/client/memory/models.js
|
2015-10-10 18:09:28 +03:00
|
|
|
*/
|
2015-11-13 22:20:45 +03:00
|
|
|
const selectSnapshot = exports.selectSnapshot = function (id) {
|
2015-10-10 18:09:28 +03:00
|
|
|
return {
|
|
|
|
type: actions.SELECT_SNAPSHOT,
|
2015-11-13 22:20:45 +03:00
|
|
|
id
|
2015-10-10 18:09:28 +03:00
|
|
|
};
|
|
|
|
};
|