chore: improve signal performance by reducing duplicate deps (#14945)

* chore: expand benchmark iterations

* chore: improve signal performance by reducing duplicate deps

* feedback

* oops
pull/14943/head
Dominic Gannaway 3 weeks ago committed by GitHub
parent deb362fe51
commit 9c20eb4815
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

@ -0,0 +1,5 @@
---
'svelte': patch
---
chore: improve signal performance by reducing duplicate deps

@ -42,15 +42,12 @@ function log_entry(signal, entry) {
const type = (signal.f & DERIVED) !== 0 ? '$derived' : '$state'; const type = (signal.f & DERIVED) !== 0 ? '$derived' : '$state';
const current_reaction = /** @type {Reaction} */ (active_reaction); const current_reaction = /** @type {Reaction} */ (active_reaction);
const status = const dirty = signal.wv > current_reaction.wv || current_reaction.wv === 0;
signal.version > current_reaction.version || current_reaction.version === 0 ? 'dirty' : 'clean';
// eslint-disable-next-line no-console // eslint-disable-next-line no-console
console.groupCollapsed( console.groupCollapsed(
`%c${type}`, `%c${type}`,
status !== 'clean' dirty ? 'color: CornflowerBlue; font-weight: bold' : 'color: grey; font-weight: bold',
? 'color: CornflowerBlue; font-weight: bold'
: 'color: grey; font-weight: bold',
typeof value === 'object' && value !== null && STATE_SYMBOL in value typeof value === 'object' && value !== null && STATE_SYMBOL in value
? snapshot(value, true) ? snapshot(value, true)
: value : value

@ -16,7 +16,7 @@ import {
set_signal_status, set_signal_status,
skip_reaction, skip_reaction,
update_reaction, update_reaction,
increment_version, increment_write_version,
set_active_effect, set_active_effect,
component_context component_context
} from '../runtime.js'; } from '../runtime.js';
@ -58,8 +58,9 @@ export function derived(fn) {
f: flags, f: flags,
fn, fn,
reactions: null, reactions: null,
rv: 0,
v: /** @type {V} */ (null), v: /** @type {V} */ (null),
version: 0, wv: 0,
parent: parent_derived ?? active_effect parent: parent_derived ?? active_effect
}; };
@ -182,7 +183,7 @@ export function update_derived(derived) {
if (!derived.equals(value)) { if (!derived.equals(value)) {
derived.v = value; derived.v = value;
derived.version = increment_version(); derived.wv = increment_write_version();
} }
} }

@ -110,7 +110,7 @@ function create_effect(type, fn, sync, push = true) {
prev: null, prev: null,
teardown: null, teardown: null,
transitions: null, transitions: null,
version: 0 wv: 0
}; };
if (DEV) { if (DEV) {

@ -12,7 +12,7 @@ import {
set_untracked_writes, set_untracked_writes,
set_signal_status, set_signal_status,
untrack, untrack,
increment_version, increment_write_version,
update_effect, update_effect,
derived_sources, derived_sources,
set_derived_sources, set_derived_sources,
@ -57,7 +57,8 @@ export function source(v, stack) {
v, v,
reactions: null, reactions: null,
equals, equals,
version: 0 rv: 0,
wv: 0
}; };
if (DEV && tracing_mode_flag) { if (DEV && tracing_mode_flag) {
@ -169,7 +170,7 @@ export function internal_set(source, value) {
if (!source.equals(value)) { if (!source.equals(value)) {
var old_value = source.v; var old_value = source.v;
source.v = value; source.v = value;
source.version = increment_version(); source.wv = increment_write_version();
if (DEV && tracing_mode_flag) { if (DEV && tracing_mode_flag) {
source.updated = get_stack('UpdatedAt'); source.updated = get_stack('UpdatedAt');

@ -4,14 +4,16 @@ export interface Signal {
/** Flags bitmask */ /** Flags bitmask */
f: number; f: number;
/** Write version */ /** Write version */
version: number; wv: number;
} }
export interface Value<V = unknown> extends Signal { export interface Value<V = unknown> extends Signal {
/** Signals that read from this signal */
reactions: null | Reaction[];
/** Equality function */ /** Equality function */
equals: Equals; equals: Equals;
/** Signals that read from this signal */
reactions: null | Reaction[];
/** Read version */
rv: number;
/** The latest value for this signal */ /** The latest value for this signal */
v: V; v: V;
/** Dev only */ /** Dev only */

@ -127,8 +127,14 @@ export function set_untracked_writes(value) {
untracked_writes = value; untracked_writes = value;
} }
/** @type {number} Used by sources and deriveds for handling updates to unowned deriveds it starts from 1 to differentiate between a created effect and a run one for tracing */ /**
let current_version = 1; * @type {number} Used by sources and deriveds for handling updates.
* Version starts from 1 so that unowned deriveds differentiate between a created effect and a run one for tracing
**/
let write_version = 1;
/** @type {number} Used to version each read of a source of derived to avoid duplicating depedencies inside a reaction */
let read_version = 0;
// If we are working with a get() chain that has no active container, // If we are working with a get() chain that has no active container,
// to prevent memory leaks, we skip adding the reaction. // to prevent memory leaks, we skip adding the reaction.
@ -168,8 +174,8 @@ export function set_dev_current_component_function(fn) {
dev_current_component_function = fn; dev_current_component_function = fn;
} }
export function increment_version() { export function increment_write_version() {
return ++current_version; return ++write_version;
} }
/** @returns {boolean} */ /** @returns {boolean} */
@ -226,7 +232,7 @@ export function check_dirtiness(reaction) {
update_derived(/** @type {Derived} */ (dependency)); update_derived(/** @type {Derived} */ (dependency));
} }
if (dependency.version > reaction.version) { if (dependency.wv > reaction.wv) {
return true; return true;
} }
} }
@ -398,6 +404,7 @@ export function update_reaction(reaction) {
skip_reaction = !is_flushing_effect && (flags & UNOWNED) !== 0; skip_reaction = !is_flushing_effect && (flags & UNOWNED) !== 0;
derived_sources = null; derived_sources = null;
component_context = reaction.ctx; component_context = reaction.ctx;
read_version++;
try { try {
var result = /** @type {Function} */ (0, reaction.fn)(); var result = /** @type {Function} */ (0, reaction.fn)();
@ -528,7 +535,7 @@ export function update_effect(effect) {
execute_effect_teardown(effect); execute_effect_teardown(effect);
var teardown = update_reaction(effect); var teardown = update_reaction(effect);
effect.teardown = typeof teardown === 'function' ? teardown : null; effect.teardown = typeof teardown === 'function' ? teardown : null;
effect.version = current_version; effect.wv = write_version;
var deps = effect.deps; var deps = effect.deps;
@ -540,7 +547,7 @@ export function update_effect(effect) {
for (let i = 0; i < deps.length; i++) { for (let i = 0; i < deps.length; i++) {
var dep = deps[i]; var dep = deps[i];
if (dep.trace_need_increase) { if (dep.trace_need_increase) {
dep.version = increment_version(); dep.wv = increment_write_version();
dep.trace_need_increase = undefined; dep.trace_need_increase = undefined;
dep.trace_v = undefined; dep.trace_v = undefined;
} }
@ -880,27 +887,29 @@ export function get(signal) {
e.state_unsafe_local_read(); e.state_unsafe_local_read();
} }
var deps = active_reaction.deps; var deps = active_reaction.deps;
if (signal.rv < read_version) {
signal.rv = read_version;
// If the signal is accessing the same dependencies in the same
// order as it did last time, increment `skipped_deps`
// rather than updating `new_deps`, which creates GC cost
if (new_deps === null && deps !== null && deps[skipped_deps] === signal) {
skipped_deps++;
} else if (new_deps === null) {
new_deps = [signal];
} else {
new_deps.push(signal);
}
// If the signal is accessing the same dependencies in the same if (
// order as it did last time, increment `skipped_deps` untracked_writes !== null &&
// rather than updating `new_deps`, which creates GC cost active_effect !== null &&
if (new_deps === null && deps !== null && deps[skipped_deps] === signal) { (active_effect.f & CLEAN) !== 0 &&
skipped_deps++; (active_effect.f & BRANCH_EFFECT) === 0 &&
} else if (new_deps === null) { untracked_writes.includes(signal)
new_deps = [signal]; ) {
} else { set_signal_status(active_effect, DIRTY);
new_deps.push(signal); schedule_effect(active_effect);
} }
if (
untracked_writes !== null &&
active_effect !== null &&
(active_effect.f & CLEAN) !== 0 &&
(active_effect.f & BRANCH_EFFECT) === 0 &&
untracked_writes.includes(signal)
) {
set_signal_status(active_effect, DIRTY);
schedule_effect(active_effect);
} }
} else if (is_derived && /** @type {Derived} */ (signal).deps === null) { } else if (is_derived && /** @type {Derived} */ (signal).deps === null) {
var derived = /** @type {Derived} */ (signal); var derived = /** @type {Derived} */ (signal);

@ -296,6 +296,7 @@ describe('signals', () => {
const destroy = effect_root(() => { const destroy = effect_root(() => {
user_effect(() => { user_effect(() => {
log.push($.get(calc)); log.push($.get(calc));
$.get(calc);
}); });
}); });
@ -306,7 +307,7 @@ describe('signals', () => {
flushSync(() => set(count, 4)); flushSync(() => set(count, 4));
flushSync(() => set(count, 0)); flushSync(() => set(count, 0));
// Ensure we're not leaking consumers // Ensure we're not leaking consumers
assert.deepEqual(count.reactions?.length, 2); assert.deepEqual(count.reactions?.length, 1);
assert.deepEqual(calc.reactions?.length, 1); assert.deepEqual(calc.reactions?.length, 1);
assert.deepEqual(log, [0, 2, 'limit', 0]); assert.deepEqual(log, [0, 2, 'limit', 0]);
destroy(); destroy();

Loading…
Cancel
Save