Skip to main content

i_slint_compiler/llr/optim_passes/
inline_expressions.rs

1// Copyright © SixtyFPS GmbH <info@slint.dev>
2// SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-Slint-Royalty-free-2.0 OR LicenseRef-Slint-Software-3.0
3
4//! Inline properties that are simple enough to be inlined
5//!
6//! If an expression does a single property access or less, it can be inlined
7//! in the calling expression
8
9use crate::expression_tree::{BuiltinFunction, ImageReference};
10use crate::llr::{CompilationUnit, EvaluationContext, Expression};
11
12const PROPERTY_ACCESS_COST: isize = 1000;
13const ALLOC_COST: isize = 700;
14const ARRAY_INDEX_COST: isize = 500;
15/// The threshold from which we consider an expression to be worth inlining.
16/// less than two allocations. (since property access usually cost one allocation)
17const INLINE_THRESHOLD: isize = ALLOC_COST * 2 - 10;
18/// Property that are used only once should almost always be inlined unless it is really expensive to compute and we want to cache the result
19const INLINE_SINGLE_THRESHOLD: isize = ALLOC_COST * 10;
20
21// The cost of an expression.
22fn expression_cost(exp: &Expression, ctx: &EvaluationContext) -> isize {
23    let mut cost = match exp {
24        Expression::StringLiteral(_) => ALLOC_COST,
25        Expression::NumberLiteral(_) => 0,
26        Expression::BoolLiteral(_) => 0,
27        Expression::KeysLiteral(_) => 0,
28        Expression::PropertyReference(_) => PROPERTY_ACCESS_COST,
29        Expression::FunctionParameterReference { .. } => return isize::MAX,
30        Expression::StoreLocalVariable { .. } => 0,
31        Expression::ReadLocalVariable { .. } => 1,
32        Expression::StructFieldAccess { .. } => 1,
33        Expression::ArrayIndex { .. } => ARRAY_INDEX_COST,
34        Expression::Cast { .. } => 0,
35        Expression::CodeBlock(_) => 0,
36        Expression::BuiltinFunctionCall { function, .. } => builtin_function_cost(function),
37        Expression::CallBackCall { callback, .. } => callback_cost(callback, ctx),
38        Expression::FunctionCall { function, .. } => callback_cost(function, ctx),
39        Expression::ItemMemberFunctionCall { function } => callback_cost(function, ctx),
40        Expression::ExtraBuiltinFunctionCall { .. } => return isize::MAX,
41        Expression::PropertyAssignment { .. } => return isize::MAX,
42        Expression::ModelDataAssignment { .. } => return isize::MAX,
43        Expression::ArrayIndexAssignment { .. } => return isize::MAX,
44        Expression::SliceIndexAssignment { .. } => return isize::MAX,
45        Expression::BinaryExpression { .. } => 1,
46        Expression::UnaryOp { .. } => 1,
47        // Avoid inlining calls to load the image from the cache, as in the worst case the image isn't cached
48        // and repeated calls will load the image over and over again. It's better to keep the image cached in the
49        // `property<image>` of the `Image` element, with the exception of embedded textures.
50        Expression::ImageReference {
51            resource_ref: ImageReference::EmbeddedTexture { .. }, ..
52        } => 1,
53        Expression::ImageReference { .. } => return isize::MAX,
54        Expression::Condition { condition, true_expr, false_expr } => {
55            return expression_cost(condition, ctx)
56                .saturating_add(
57                    expression_cost(true_expr, ctx).max(expression_cost(false_expr, ctx)),
58                )
59                .saturating_add(10);
60        }
61        // Never inline an array because it is a model and when shared it needs to keep its identity
62        // (cf #5249)  (otherwise it would be `ALLOC_COST`)
63        Expression::Array { .. } => return isize::MAX,
64        Expression::Struct { .. } => 1,
65        Expression::EasingCurve(_) => 1,
66        Expression::LinearGradient { .. } => ALLOC_COST,
67        Expression::RadialGradient { .. } => ALLOC_COST,
68        Expression::ConicGradient { .. } => ALLOC_COST,
69        Expression::EnumerationValue(_) => 0,
70        Expression::LayoutCacheAccess { .. } => PROPERTY_ACCESS_COST,
71        Expression::GridRepeaterCacheAccess { .. } => PROPERTY_ACCESS_COST,
72        Expression::WithLayoutItemInfo { .. } => return isize::MAX,
73        Expression::WithFlexboxLayoutItemInfo { .. } => return isize::MAX,
74        Expression::WithGridInputData { .. } => return isize::MAX,
75        Expression::MinMax { .. } => 10,
76        Expression::EmptyComponentFactory => 10,
77        Expression::TranslationReference { .. } => PROPERTY_ACCESS_COST + 2 * ALLOC_COST,
78    };
79
80    exp.visit(|e| cost = cost.saturating_add(expression_cost(e, ctx)));
81
82    cost
83}
84
85fn callback_cost(_callback: &crate::llr::MemberReference, _ctx: &EvaluationContext) -> isize {
86    // TODO: lookup the callback and find out what it does
87    isize::MAX
88}
89
90fn builtin_function_cost(function: &BuiltinFunction) -> isize {
91    match function {
92        BuiltinFunction::GetWindowScaleFactor => PROPERTY_ACCESS_COST,
93        BuiltinFunction::GetWindowDefaultFontSize => PROPERTY_ACCESS_COST,
94        BuiltinFunction::AnimationTick => PROPERTY_ACCESS_COST,
95        BuiltinFunction::Debug => isize::MAX,
96        BuiltinFunction::Mod => 10,
97        BuiltinFunction::Round => 10,
98        BuiltinFunction::Ceil => 10,
99        BuiltinFunction::Floor => 10,
100        BuiltinFunction::Abs => 10,
101        BuiltinFunction::Sqrt => 10,
102        BuiltinFunction::Cos => 10,
103        BuiltinFunction::Sin => 10,
104        BuiltinFunction::Tan => 10,
105        BuiltinFunction::ACos => 10,
106        BuiltinFunction::ASin => 10,
107        BuiltinFunction::ATan => 10,
108        BuiltinFunction::ATan2 => 10,
109        BuiltinFunction::Log => 10,
110        BuiltinFunction::Ln => 10,
111        BuiltinFunction::Pow => 10,
112        BuiltinFunction::Exp => 10,
113        BuiltinFunction::ToFixed => ALLOC_COST,
114        BuiltinFunction::ToPrecision => ALLOC_COST,
115        BuiltinFunction::SetFocusItem | BuiltinFunction::ClearFocusItem => isize::MAX,
116        BuiltinFunction::ShowPopupWindow
117        | BuiltinFunction::ClosePopupWindow
118        | BuiltinFunction::ShowPopupMenu
119        | BuiltinFunction::ShowPopupMenuInternal => isize::MAX,
120        BuiltinFunction::SetSelectionOffsets => isize::MAX,
121        BuiltinFunction::ItemFontMetrics => PROPERTY_ACCESS_COST,
122        BuiltinFunction::StringToFloat => 50,
123        BuiltinFunction::StringIsFloat => 50,
124        BuiltinFunction::StringIsEmpty => 50,
125        BuiltinFunction::StringCharacterCount => 50,
126        BuiltinFunction::StringToLowercase => ALLOC_COST,
127        BuiltinFunction::StringToUppercase => ALLOC_COST,
128        BuiltinFunction::KeysToString => ALLOC_COST,
129        BuiltinFunction::ColorRgbaStruct => 50,
130        BuiltinFunction::ColorHsvaStruct => 50,
131        BuiltinFunction::ColorOklchStruct => 50,
132        BuiltinFunction::ColorBrighter => 50,
133        BuiltinFunction::ColorDarker => 50,
134        BuiltinFunction::ColorTransparentize => 50,
135        BuiltinFunction::ColorMix => 50,
136        BuiltinFunction::ColorWithAlpha => 50,
137        BuiltinFunction::ImageSize => 50,
138        BuiltinFunction::ArrayLength => 50,
139        BuiltinFunction::Rgb => 50,
140        BuiltinFunction::Hsv => 50,
141        BuiltinFunction::Oklch => 50,
142        BuiltinFunction::ImplicitLayoutInfo(_) => isize::MAX,
143        BuiltinFunction::ItemAbsolutePosition => isize::MAX,
144        BuiltinFunction::RegisterCustomFontByPath => isize::MAX,
145        BuiltinFunction::RegisterCustomFontByMemory => isize::MAX,
146        BuiltinFunction::RegisterBitmapFont => isize::MAX,
147        BuiltinFunction::ColorScheme => PROPERTY_ACCESS_COST,
148        BuiltinFunction::AccentColor => PROPERTY_ACCESS_COST,
149        BuiltinFunction::SupportsNativeMenuBar => 10,
150        BuiltinFunction::SetupMenuBar => isize::MAX,
151        BuiltinFunction::MonthDayCount => isize::MAX,
152        BuiltinFunction::MonthOffset => isize::MAX,
153        BuiltinFunction::FormatDate => isize::MAX,
154        BuiltinFunction::DateNow => isize::MAX,
155        BuiltinFunction::ValidDate => isize::MAX,
156        BuiltinFunction::ParseDate => isize::MAX,
157        BuiltinFunction::SetTextInputFocused => PROPERTY_ACCESS_COST,
158        BuiltinFunction::TextInputFocused => PROPERTY_ACCESS_COST,
159        BuiltinFunction::Translate => 2 * ALLOC_COST + PROPERTY_ACCESS_COST,
160        BuiltinFunction::Use24HourFormat => 2 * ALLOC_COST + PROPERTY_ACCESS_COST,
161        BuiltinFunction::UpdateTimers => 10,
162        BuiltinFunction::DetectOperatingSystem => 10,
163        BuiltinFunction::StartTimer => 10,
164        BuiltinFunction::StopTimer => 10,
165        BuiltinFunction::RestartTimer => 10,
166        BuiltinFunction::ParseMarkdown => isize::MAX,
167        BuiltinFunction::StringToStyledText => ALLOC_COST,
168        BuiltinFunction::OpenUrl => isize::MAX,
169    }
170}
171
172pub fn inline_simple_expressions(root: &CompilationUnit) {
173    root.for_each_expression(&mut |e, ctx| {
174        inline_simple_expressions_in_expression(&mut e.borrow_mut(), ctx)
175    })
176}
177
178fn inline_simple_expressions_in_expression(expr: &mut Expression, ctx: &EvaluationContext) {
179    if let Expression::PropertyReference(prop) = expr {
180        let prop_info = ctx.property_info(prop);
181        if prop_info.analysis.as_ref().is_some_and(|a| !a.is_set && !a.is_set_externally) {
182            if let Some((binding, map)) = prop_info.binding {
183                if binding.animation.is_none()
184                    // State info binding are special and the binding cannot be inlined or used.
185                    && !binding.is_state_info
186                {
187                    let mapped_ctx = map.map_context(ctx);
188                    let cost = expression_cost(&binding.expression.borrow(), &mapped_ctx);
189                    let use_count = binding.use_count.get();
190                    debug_assert!(
191                        use_count > 0,
192                        "We use a property and its count is zero: {}",
193                        crate::llr::pretty_print::DisplayPropertyRef(prop, ctx)
194                    );
195                    if cost <= INLINE_THRESHOLD
196                        || (use_count == 1 && cost <= INLINE_SINGLE_THRESHOLD)
197                    {
198                        // Perform inlining
199                        *expr = binding.expression.borrow().clone();
200                        map.map_expression(expr);
201                        // adjust use count
202                        binding.use_count.set(use_count - 1);
203                        if let Some(use_count) = prop_info.use_count {
204                            use_count.set(use_count.get() - 1);
205                        }
206                        adjust_use_count(expr, ctx, 1);
207                        if use_count == 1 {
208                            adjust_use_count(&binding.expression.borrow(), &mapped_ctx, -1);
209                            binding.expression.replace(Expression::CodeBlock(Vec::new()));
210                        }
211                    }
212                }
213            } else if let Some(use_count) = prop_info.use_count
214                && let Some(e) = Expression::default_value_for_type(&prop_info.ty)
215            {
216                use_count.set(use_count.get() - 1);
217                *expr = e;
218            }
219        }
220    };
221
222    expr.visit_mut(|e| inline_simple_expressions_in_expression(e, ctx));
223}
224
225fn adjust_use_count(expr: &Expression, ctx: &EvaluationContext, adjust: isize) {
226    expr.visit_property_references(ctx, &mut |p, ctx| {
227        let prop_info = ctx.property_info(p);
228        if let Some(use_count) = prop_info.use_count {
229            use_count.set(use_count.get().checked_add_signed(adjust).unwrap());
230        }
231        if let Some((binding, _)) = prop_info.binding {
232            let use_count = binding.use_count.get().checked_add_signed(adjust).unwrap();
233            binding.use_count.set(use_count);
234        }
235    });
236}