Consolidate measure cache for layout nodes and leaf nodes
Summary: Re-use advanced changing logic of leaf nodes in layout nodes. //benchmark:benchmark nested flex test shows a ~15% improvement in layout performance! Reviewed By: astreet Differential Revision: D4957570 fbshipit-source-id: 5c5f39b67bd3f72d92b939ecee2d9a46c80b583f
This commit is contained in:
committed by
Facebook Github Bot
parent
7c324f058c
commit
77c05b5ff6
40
yoga/Yoga.c
40
yoga/Yoga.c
@@ -3116,26 +3116,18 @@ bool YGLayoutNodeInternal(const YGNodeRef node,
|
||||
layout->cachedLayout.computedHeight = -1;
|
||||
}
|
||||
|
||||
YGCachedMeasurement *cachedResults = NULL;
|
||||
|
||||
// Determine whether the results are already cached. We maintain a separate
|
||||
// cache for layouts and measurements. A layout operation modifies the
|
||||
// positions
|
||||
// and dimensions for nodes in the subtree. The algorithm assumes that each
|
||||
// node
|
||||
// gets layed out a maximum of one time per tree layout, but multiple
|
||||
// measurements
|
||||
// may be required to resolve all of the flex dimensions.
|
||||
// We handle nodes with measure functions specially here because they are the
|
||||
// most
|
||||
// expensive to measure, so it's worth avoiding redundant measurements if at
|
||||
// all possible.
|
||||
if (node->measure) {
|
||||
// positions and dimensions for nodes in the subtree. The algorithm assumes that each
|
||||
// node gets layed out a maximum of one time per tree layout, but multiple
|
||||
// measurements may be required to resolve all of the flex dimensions.
|
||||
|
||||
YGCachedMeasurement *cachedResults = NULL;
|
||||
const float marginAxisRow = YGNodeMarginForAxis(node, YGFlexDirectionRow, parentWidth);
|
||||
const float marginAxisColumn = YGNodeMarginForAxis(node, YGFlexDirectionColumn, parentWidth);
|
||||
|
||||
// First, try to use the layout cache.
|
||||
if (YGNodeCanUseCachedMeasurement(widthMeasureMode,
|
||||
if ((node->measure || performLayout) && YGNodeCanUseCachedMeasurement(widthMeasureMode,
|
||||
availableWidth,
|
||||
heightMeasureMode,
|
||||
availableHeight,
|
||||
@@ -3148,7 +3140,7 @@ bool YGLayoutNodeInternal(const YGNodeRef node,
|
||||
marginAxisRow,
|
||||
marginAxisColumn)) {
|
||||
cachedResults = &layout->cachedLayout;
|
||||
} else {
|
||||
} else if (node->measure || !performLayout) {
|
||||
// Try to use the measurement cache.
|
||||
for (uint32_t i = 0; i < layout->nextCachedMeasurementsIndex; i++) {
|
||||
if (YGNodeCanUseCachedMeasurement(widthMeasureMode,
|
||||
@@ -3168,24 +3160,6 @@ bool YGLayoutNodeInternal(const YGNodeRef node,
|
||||
}
|
||||
}
|
||||
}
|
||||
} else if (performLayout) {
|
||||
if (YGFloatsEqual(layout->cachedLayout.availableWidth, availableWidth) &&
|
||||
YGFloatsEqual(layout->cachedLayout.availableHeight, availableHeight) &&
|
||||
layout->cachedLayout.widthMeasureMode == widthMeasureMode &&
|
||||
layout->cachedLayout.heightMeasureMode == heightMeasureMode) {
|
||||
cachedResults = &layout->cachedLayout;
|
||||
}
|
||||
} else {
|
||||
for (uint32_t i = 0; i < layout->nextCachedMeasurementsIndex; i++) {
|
||||
if (YGFloatsEqual(layout->cachedMeasurements[i].availableWidth, availableWidth) &&
|
||||
YGFloatsEqual(layout->cachedMeasurements[i].availableHeight, availableHeight) &&
|
||||
layout->cachedMeasurements[i].widthMeasureMode == widthMeasureMode &&
|
||||
layout->cachedMeasurements[i].heightMeasureMode == heightMeasureMode) {
|
||||
cachedResults = &layout->cachedMeasurements[i];
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (!needToVisitNode && cachedResults != NULL) {
|
||||
layout->measuredDimensions[YGDimensionWidth] = cachedResults->computedWidth;
|
||||
|
Reference in New Issue
Block a user