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::PropertyReference(_) => PROPERTY_ACCESS_COST,
28        Expression::FunctionParameterReference { .. } => return isize::MAX,
29        Expression::StoreLocalVariable { .. } => 0,
30        Expression::ReadLocalVariable { .. } => 1,
31        Expression::StructFieldAccess { .. } => 1,
32        Expression::ArrayIndex { .. } => ARRAY_INDEX_COST,
33        Expression::Cast { .. } => 0,
34        Expression::CodeBlock(_) => 0,
35        Expression::BuiltinFunctionCall { function, .. } => builtin_function_cost(function),
36        Expression::CallBackCall { callback, .. } => callback_cost(callback, ctx),
37        Expression::FunctionCall { function, .. } => callback_cost(function, ctx),
38        Expression::ItemMemberFunctionCall { function } => callback_cost(function, ctx),
39        Expression::ExtraBuiltinFunctionCall { .. } => return isize::MAX,
40        Expression::PropertyAssignment { .. } => return isize::MAX,
41        Expression::ModelDataAssignment { .. } => return isize::MAX,
42        Expression::ArrayIndexAssignment { .. } => return isize::MAX,
43        Expression::BinaryExpression { .. } => 1,
44        Expression::UnaryOp { .. } => 1,
45        // Avoid inlining calls to load the image from the cache, as in the worst case the image isn't cached
46        // and repeated calls will load the image over and over again. It's better to keep the image cached in the
47        // `property<image>` of the `Image` element, with the exception of embedded textures.
48        Expression::ImageReference {
49            resource_ref: ImageReference::EmbeddedTexture { .. }, ..
50        } => 1,
51        Expression::ImageReference { .. } => return isize::MAX,
52        Expression::Condition { condition, true_expr, false_expr } => {
53            return expression_cost(condition, ctx)
54                .saturating_add(
55                    expression_cost(true_expr, ctx).max(expression_cost(false_expr, ctx)),
56                )
57                .saturating_add(10);
58        }
59        // Never inline an array because it is a model and when shared it needs to keep its identity
60        // (cf #5249)  (otherwise it would be `ALLOC_COST`)
61        Expression::Array { .. } => return isize::MAX,
62        Expression::Struct { .. } => 1,
63        Expression::EasingCurve(_) => 1,
64        Expression::LinearGradient { .. } => ALLOC_COST,
65        Expression::RadialGradient { .. } => ALLOC_COST,
66        Expression::ConicGradient { .. } => ALLOC_COST,
67        Expression::EnumerationValue(_) => 0,
68        Expression::LayoutCacheAccess { .. } => PROPERTY_ACCESS_COST,
69        Expression::BoxLayoutFunction { .. } => return isize::MAX,
70        Expression::ComputeDialogLayoutCells { .. } => return isize::MAX,
71        Expression::MinMax { .. } => 10,
72        Expression::EmptyComponentFactory => 10,
73        Expression::TranslationReference { .. } => PROPERTY_ACCESS_COST + 2 * ALLOC_COST,
74    };
75
76    exp.visit(|e| cost = cost.saturating_add(expression_cost(e, ctx)));
77
78    cost
79}
80
81fn callback_cost(_callback: &crate::llr::PropertyReference, _ctx: &EvaluationContext) -> isize {
82    // TODO: lookup the callback and find out what it does
83    isize::MAX
84}
85
86fn builtin_function_cost(function: &BuiltinFunction) -> isize {
87    match function {
88        BuiltinFunction::GetWindowScaleFactor => PROPERTY_ACCESS_COST,
89        BuiltinFunction::GetWindowDefaultFontSize => PROPERTY_ACCESS_COST,
90        BuiltinFunction::AnimationTick => PROPERTY_ACCESS_COST,
91        BuiltinFunction::Debug => isize::MAX,
92        BuiltinFunction::Mod => 10,
93        BuiltinFunction::Round => 10,
94        BuiltinFunction::Ceil => 10,
95        BuiltinFunction::Floor => 10,
96        BuiltinFunction::Abs => 10,
97        BuiltinFunction::Sqrt => 10,
98        BuiltinFunction::Cos => 10,
99        BuiltinFunction::Sin => 10,
100        BuiltinFunction::Tan => 10,
101        BuiltinFunction::ACos => 10,
102        BuiltinFunction::ASin => 10,
103        BuiltinFunction::ATan => 10,
104        BuiltinFunction::ATan2 => 10,
105        BuiltinFunction::Log => 10,
106        BuiltinFunction::Ln => 10,
107        BuiltinFunction::Pow => 10,
108        BuiltinFunction::Exp => 10,
109        BuiltinFunction::ToFixed => ALLOC_COST,
110        BuiltinFunction::ToPrecision => ALLOC_COST,
111        BuiltinFunction::SetFocusItem | BuiltinFunction::ClearFocusItem => isize::MAX,
112        BuiltinFunction::ShowPopupWindow
113        | BuiltinFunction::ClosePopupWindow
114        | BuiltinFunction::ShowPopupMenu
115        | BuiltinFunction::ShowPopupMenuInternal => isize::MAX,
116        BuiltinFunction::SetSelectionOffsets => isize::MAX,
117        BuiltinFunction::ItemFontMetrics => PROPERTY_ACCESS_COST,
118        BuiltinFunction::StringToFloat => 50,
119        BuiltinFunction::StringIsFloat => 50,
120        BuiltinFunction::StringIsEmpty => 50,
121        BuiltinFunction::StringCharacterCount => 50,
122        BuiltinFunction::StringToLowercase => ALLOC_COST,
123        BuiltinFunction::StringToUppercase => ALLOC_COST,
124        BuiltinFunction::ColorRgbaStruct => 50,
125        BuiltinFunction::ColorHsvaStruct => 50,
126        BuiltinFunction::ColorBrighter => 50,
127        BuiltinFunction::ColorDarker => 50,
128        BuiltinFunction::ColorTransparentize => 50,
129        BuiltinFunction::ColorMix => 50,
130        BuiltinFunction::ColorWithAlpha => 50,
131        BuiltinFunction::ImageSize => 50,
132        BuiltinFunction::ArrayLength => 50,
133        BuiltinFunction::Rgb => 50,
134        BuiltinFunction::Hsv => 50,
135        BuiltinFunction::ImplicitLayoutInfo(_) => isize::MAX,
136        BuiltinFunction::ItemAbsolutePosition => isize::MAX,
137        BuiltinFunction::RegisterCustomFontByPath => isize::MAX,
138        BuiltinFunction::RegisterCustomFontByMemory => isize::MAX,
139        BuiltinFunction::RegisterBitmapFont => isize::MAX,
140        BuiltinFunction::ColorScheme => PROPERTY_ACCESS_COST,
141        BuiltinFunction::SupportsNativeMenuBar => 10,
142        BuiltinFunction::SetupMenuBar => isize::MAX,
143        BuiltinFunction::MonthDayCount => isize::MAX,
144        BuiltinFunction::MonthOffset => isize::MAX,
145        BuiltinFunction::FormatDate => isize::MAX,
146        BuiltinFunction::DateNow => isize::MAX,
147        BuiltinFunction::ValidDate => isize::MAX,
148        BuiltinFunction::ParseDate => isize::MAX,
149        BuiltinFunction::SetTextInputFocused => PROPERTY_ACCESS_COST,
150        BuiltinFunction::TextInputFocused => PROPERTY_ACCESS_COST,
151        BuiltinFunction::Translate => 2 * ALLOC_COST + PROPERTY_ACCESS_COST,
152        BuiltinFunction::Use24HourFormat => 2 * ALLOC_COST + PROPERTY_ACCESS_COST,
153        BuiltinFunction::UpdateTimers => 10,
154        BuiltinFunction::DetectOperatingSystem => 10,
155        BuiltinFunction::StartTimer => 10,
156        BuiltinFunction::StopTimer => 10,
157        BuiltinFunction::RestartTimer => 10,
158    }
159}
160
161pub fn inline_simple_expressions(root: &CompilationUnit) {
162    root.for_each_expression(&mut |e, ctx| {
163        inline_simple_expressions_in_expression(&mut e.borrow_mut(), ctx)
164    })
165}
166
167fn inline_simple_expressions_in_expression(expr: &mut Expression, ctx: &EvaluationContext) {
168    if let Expression::PropertyReference(prop) = expr {
169        let prop_info = ctx.property_info(prop);
170        if prop_info.analysis.as_ref().is_some_and(|a| !a.is_set && !a.is_set_externally) {
171            if let Some((binding, map)) = prop_info.binding {
172                if binding.animation.is_none()
173                    // State info binding are special and the binding cannot be inlined or used.
174                    && !binding.is_state_info
175                {
176                    let mapped_ctx = map.map_context(ctx);
177                    let cost = expression_cost(&binding.expression.borrow(), &mapped_ctx);
178                    let use_count = binding.use_count.get();
179                    debug_assert!(
180                        use_count > 0,
181                        "We use a property and its count is zero: {}",
182                        crate::llr::pretty_print::DisplayPropertyRef(prop, ctx)
183                    );
184                    if cost <= INLINE_THRESHOLD
185                        || (use_count == 1 && cost <= INLINE_SINGLE_THRESHOLD)
186                    {
187                        // Perform inlining
188                        *expr = binding.expression.borrow().clone();
189                        map.map_expression(expr);
190                        // adjust use count
191                        binding.use_count.set(use_count - 1);
192                        if let Some(prop_decl) = prop_info.property_decl {
193                            prop_decl.use_count.set(prop_decl.use_count.get() - 1);
194                        }
195                        adjust_use_count(expr, ctx, 1);
196                        if use_count == 1 {
197                            adjust_use_count(&binding.expression.borrow(), &mapped_ctx, -1);
198                            binding.expression.replace(Expression::CodeBlock(vec![]));
199                        }
200                    }
201                }
202            } else if let Some(prop_decl) = prop_info.property_decl {
203                if let Some(e) = Expression::default_value_for_type(&prop_decl.ty) {
204                    prop_decl.use_count.set(prop_decl.use_count.get() - 1);
205                    *expr = e;
206                }
207            }
208        }
209    };
210
211    expr.visit_mut(|e| inline_simple_expressions_in_expression(e, ctx));
212}
213
214fn adjust_use_count(expr: &Expression, ctx: &EvaluationContext, adjust: isize) {
215    expr.visit_property_references(ctx, &mut |p, ctx| {
216        let prop_info = ctx.property_info(p);
217        if let Some(property_decl) = prop_info.property_decl {
218            property_decl
219                .use_count
220                .set(property_decl.use_count.get().checked_add_signed(adjust).unwrap());
221        }
222        if let Some((binding, _)) = prop_info.binding {
223            let use_count = binding.use_count.get().checked_add_signed(adjust).unwrap();
224            binding.use_count.set(use_count);
225        }
226    });
227}