Skip to content

Commit

Permalink
Merge pull request #5247 from voxel51/fix/transfer-bitmaps-back
Browse files Browse the repository at this point in the history
sample update / overlay recoloring performance optimization
  • Loading branch information
sashankaryal authored Dec 17, 2024
2 parents f627b82 + e22b56e commit 35f15b2
Show file tree
Hide file tree
Showing 6 changed files with 76 additions and 18 deletions.
69 changes: 64 additions & 5 deletions app/packages/looker/src/lookers/abstract.ts
Original file line number Diff line number Diff line change
Expand Up @@ -23,7 +23,7 @@ import {
import { Events } from "../elements/base";
import { COMMON_SHORTCUTS, LookerElement } from "../elements/common";
import { ClassificationsOverlay, loadOverlays } from "../overlays";
import { CONTAINS, Overlay } from "../overlays/base";
import { CONTAINS, LabelMask, Overlay } from "../overlays/base";
import processOverlays from "../processOverlays";
import {
BaseState,
Expand Down Expand Up @@ -515,7 +515,44 @@ export abstract class AbstractLooker<
abstract updateOptions(options: Partial<State["options"]>): void;

updateSample(sample: Sample) {
this.loadSample(sample);
// collect any mask targets array buffer that overlays might have
// we'll transfer that to the worker instead of copying it
const arrayBuffers: ArrayBuffer[] = [];

for (const overlay of this.pluckedOverlays ?? []) {
let overlayData: LabelMask = null;

if ("mask" in overlay.label) {
overlayData = overlay.label.mask as LabelMask;
} else if ("map" in overlay.label) {
overlayData = overlay.label.map as LabelMask;
}

const buffer = overlayData?.data?.buffer;

if (!buffer) {
continue;
}

// check for detached buffer (happens if user is switching colors too fast)
// note: ArrayBuffer.prototype.detached is a new browser API
if (typeof buffer.detached !== "undefined") {
if (buffer.detached) {
// most likely sample is already being processed, skip update
return;
} else {
arrayBuffers.push(buffer);
}
} else if (buffer.byteLength) {
// hope we don't run into this edge case (old browser)
// sometimes detached buffers have bytelength > 0
// if we run into this case, we'll just attempt to transfer the buffer
// might get a DataCloneError if user is switching colors too fast
arrayBuffers.push(buffer);
}
}

this.loadSample(sample, arrayBuffers.flat());
}

getSample(): Promise<Sample> {
Expand Down Expand Up @@ -698,13 +735,22 @@ export abstract class AbstractLooker<
);
}

private loadSample(sample: Sample) {
protected cleanOverlays() {
for (const overlay of this.sampleOverlays ?? []) {
overlay.cleanup();
}
}

private loadSample(sample: Sample, transfer: Transferable[] = []) {
const messageUUID = uuid();

const labelsWorker = getLabelsWorker();

const listener = ({ data: { sample, coloring, uuid } }) => {
if (uuid === messageUUID) {
// we paint overlays again, so cleanup the old ones
// this helps prevent memory leaks from, for instance, dangling ImageBitmaps
this.cleanOverlays();
this.sample = sample;
this.state.options.coloring = coloring;
this.loadOverlays(sample);
Expand All @@ -719,7 +765,7 @@ export abstract class AbstractLooker<

labelsWorker.addEventListener("message", listener);

labelsWorker.postMessage({
const workerArgs = {
sample: sample as ProcessSample["sample"],
method: "processSample",
coloring: this.state.options.coloring,
Expand All @@ -730,7 +776,20 @@ export abstract class AbstractLooker<
sources: this.state.config.sources,
schema: this.state.config.fieldSchema,
uuid: messageUUID,
} as ProcessSample);
} as ProcessSample;

try {
labelsWorker.postMessage(workerArgs, transfer);
} catch (error) {
// rarely we'll get a DataCloneError
// if one of the buffers is detached and we didn't catch it
// try again without transferring the buffers (copying them)
if (error.name === "DataCloneError") {
labelsWorker.postMessage(workerArgs);
} else {
throw error;
}
}
}
}

Expand Down
3 changes: 2 additions & 1 deletion app/packages/looker/src/overlays/base.ts
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,7 @@ export interface Overlay<State extends Partial<BaseState>> {
draw(ctx: CanvasRenderingContext2D, state: State): void;
isShown(state: Readonly<State>): boolean;
field?: string;
label?: BaseLabel;
containsPoint(state: Readonly<State>): CONTAINS;
getMouseDistance(state: Readonly<State>): number;
getPointInfo(state: Readonly<State>): any;
Expand All @@ -82,7 +83,7 @@ export abstract class CoordinateOverlay<
> implements Overlay<State>
{
readonly field: string;
protected label: Label;
readonly label: Label;

constructor(field: string, label: Label) {
this.field = field;
Expand Down
5 changes: 1 addition & 4 deletions app/packages/looker/src/overlays/detection.ts
Original file line number Diff line number Diff line change
Expand Up @@ -262,10 +262,7 @@ export default class DetectionOverlay<
}

public cleanup(): void {
if (this.label.mask?.bitmap) {
this.label.mask?.bitmap.close();
this.label.mask.bitmap = null;
}
this.label.mask?.bitmap?.close();
}
}

Expand Down
6 changes: 2 additions & 4 deletions app/packages/looker/src/overlays/heatmap.ts
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ export default class HeatmapOverlay<State extends BaseState>
implements Overlay<State>
{
readonly field: string;
private label: HeatmapLabel;
readonly label: HeatmapLabel;
private targets?: TypedArray;
private readonly range: [number, number];

Expand Down Expand Up @@ -207,9 +207,7 @@ export default class HeatmapOverlay<State extends BaseState>
}

public cleanup(): void {
if (this.label.map?.bitmap) {
this.label.map?.bitmap.close();
}
this.label.map?.bitmap?.close();
}
}

Expand Down
6 changes: 2 additions & 4 deletions app/packages/looker/src/overlays/segmentation.ts
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ export default class SegmentationOverlay<State extends BaseState>
implements Overlay<State>
{
readonly field: string;
private label: SegmentationLabel;
readonly label: SegmentationLabel;
private targets?: TypedArray;

private isRgbMaskTargets = false;
Expand Down Expand Up @@ -262,9 +262,7 @@ export default class SegmentationOverlay<State extends BaseState>
}

public cleanup(): void {
if (this.label.mask?.bitmap) {
this.label.mask?.bitmap.close();
}
this.label.mask?.bitmap?.close();
}
}

Expand Down
5 changes: 5 additions & 0 deletions app/packages/looker/src/worker/disk-overlay-decoder.ts
Original file line number Diff line number Diff line change
Expand Up @@ -61,6 +61,11 @@ export const decodeOverlayOnDisk = async (
) {
const height = label[overlayField].bitmap.height;
const width = label[overlayField].bitmap.width;

// close the copied bitmap
label[overlayField].bitmap.close();
label[overlayField].bitmap = null;

label[overlayField].image = new ArrayBuffer(height * width * 4);
label[overlayField].bitmap.close();
label[overlayField].bitmap = null;
Expand Down

0 comments on commit 35f15b2

Please sign in to comment.