diff --git a/_data/authors.yaml b/_data/authors.yaml index 68819504be..4362e17221 100644 --- a/_data/authors.yaml +++ b/_data/authors.yaml @@ -493,3 +493,10 @@ fbricon: job_title: "Principal Software Engineer" twitter: "fbricon" bio: "Fred has been working in the Dev Tools team at Red Hat, since 2011, where he develops IDE extensions for Eclipse, VS Code and IntelliJ IDEA." +zakkak: + name: "Foivos Zakkak" + email: "foivos@zakkak.net" + emailhash: "3772ed72c143c2f0fe110d1a4124a46a" + job_title: "Software Engineer" + twitter: "zakkak" + bio: "Software Engineer at Red Hat on Mandrel and other projects." diff --git a/_posts/2023-11-02-mandrel-23-0-image-size-increase.md b/_posts/2023-11-02-mandrel-23-0-image-size-increase.md new file mode 100644 index 0000000000..892ff60336 --- /dev/null +++ b/_posts/2023-11-02-mandrel-23-0-image-size-increase.md @@ -0,0 +1,570 @@ +--- +layout: post +title: Exploring why native executables produced with Mandrel 23.0 are bigger than those produced with Mandrel 22.3 +date: 2023-11-02 +tags: native graalvm mandrel metrics +synopsis: A comprehensive analysis attributing binary size increase to specific changes in Mandrel's code base. +author: zakkak +--- + +Starting with Quarkus 3.2 the default Mandrel version was updated from 22.3 to 23.0. + +This update brought a number of bugfixes as well as new features like: + +1. Better support for profiling and debugging using `perf` and `gdb`. +2. Finer control over the monitoring features included in the native executable. +3. Support for more JFR events. + +However, it also brought an unwanted side effect. +The native executables produced with Mandrel 23.0 are bigger than the ones produced with Mandrel 22.3. +To better understand why that happens we perform a thorough analysis to attribute the size increase to specific changes in Mandrel's code base. + +## TL;DR + +According to our analysis the binary size increase is attributed to three distinct changes, all of which are well justified: + +1. [Skipping constant folding of reflection methods with side effects](https://github.com/oracle/graal/pull/5156). +2. [Reducing the number of stores that are executed by the serial GC write barriers to improve performance by reducing the number of cache misses](https://github.com/oracle/graal/pull/5330). +3. [Enabling code alignment to compensate for the performance penalty of Intel's jump conditional code erratum](https://github.com/oracle/graal/commit/4de58f1b3c484213951622c03d74f3435a20c4ef#diff-991a434bbfc9a6af5514e4609380d5fbfe7618585d5b1b3f11fa2a7431ca7ab0L1388-R1388). + +## Better understanding what is different between the generated native executables + +The first step in our analysis is to understand where the binary size increase comes from. +Usually such an increase is attributed to one of the following: + +1. More code being generated, due to more code becoming reachable. +2. More code being generated, due to more aggressive inlining. +3. More data being stored in the image heap, due to more objects being reachable. +4. More data being stored in the image heap, due to more types being registered for reflection thus requiring more code metadata to be stored. + +To perform the analysis we use the [Quarkus startstop test](https://github.com/quarkus-qe/quarkus-startstop) (specifically commit `a8bae846881607e376c7c8a96116b6b50ee50b70`) which generates, starts, tests, and stops small Quarkus applications and measures various time-related metrics (e.g. time-to-first-OK-request) and memory usage. +We get the test with: + +```shell +git clone https://github.com/quarkus-qe/quarkus-startstop +cd quarkus-startstop +git checkout a8bae846881607e376c7c8a96116b6b50ee50b70 +``` + +and build it with: + +```shell +mvn clean package -Pnative -Dquarkus.version=3.2.6.Final\ + -Dquarkus.native.builder-image=quay.io/quarkus/ubi-quarkus-mandrel-builder-image:jdk-17 +``` + +changing the builder image tag to `22.3-java17` for building with Mandrel 22.3. + +Looking at the build output (generated by Quarkus in `target/my-app-native-image-sources/my-app-build-output-stats.json`) the main differences between the two builds are in the following metrics: + +| Mandrel version | 22.3.3.1 | 23.0.1.2 | Increase % +| --- | --- | --- | --- | +| Image Heap Size | 28807168 | 29499392 | 2.4 +| Image Code Area | 27680208 | 29625424 | 7 +| Total Image Size | 56826648 | 59467728 | 4.6 +| Classes registered for reflection | 645 | 4317 | 570 + +Hinting that any of the reasons 1-4 mentioned above is possible. + +### Dashboards + +GraalVM and Mandrel provide the `-H:+DashboardAll` and `-H:+DashboardJson` flags that can be used to generate dashboards that contain more information about the generated native executable. +The resulting dashboard contains a number of metrics and looks like this: + +```json +{ + "points-to": { + "type-flows": [ + ... + ] + }, + "code-breakdown": { + "code-size": [ + { + "name": "io.smallrye.mutiny.CompositeException.getFirstOrFail(Throwable[]) Throwable", + "size": 575 + }, + ... + ] + }, + "heap-breakdown": { + "heap-size": [ + { + "name": "Lio/vertx/core/impl/VerticleManager$$Lambda$bf09d38f5d19578a0d041ffd0a524c1cbe1843df;", + "size": 24, + "count": 1 + }, + ... + ] + } +} +``` + +Using the aforementioned flags we generate dashboards using both Mandrel 22.3 and 23.0 and compare the results. + +To generate the dashboards using Mandrel 23.0 we use the following command: + +```shell +mvn package -Pnative \ + -Dquarkus.version=3.2.6.Final \ + -Dquarkus.native.builder-image=quay.io/quarkus/ubi-quarkus-mandrel-builder-image:jdk-17 \ + -Dquarkus.native.additional-build-args=-H:+DashboardAll,-H:+DashboardJson,-H:DashboardDump=path/to/23.0.dashboard.json +``` + +Similarly to generate the dashboards using Mandrel 22.3 we use the following command: + +```shell +mvn package -Pnative \ + -Dquarkus.version=3.2.6.Final \ + -Dquarkus.native.builder-image=quay.io/quarkus/ubi-quarkus-mandrel-builder-image:22.3-java17 \ + -Dquarkus.native.additional-build-args=-H:+DashboardAll,-H:+DashboardJson,-H:DashboardDump=path/to/22.3.dashboard.json +``` + +Note: Make sure to change `path/to/` to the path where you would like the dashboard json files to be stored, each file is about 370MB big. + +### Analyzing and visualizing the data + +To process the data from the dashboards we used a Jupyter notebook, like we did to create this article. +To grab the notebook follow [this link](/assets/examples/posts/mandrel-23-0-image-size-increase/quarkus-size-22-3-23-0.ipynb). + +For those willing to use a spreadsheet instead, a CSV file can be created to facilitate the analysis in a spreadsheet. +E.g. using `jq`: + +```shell +jq -r -s '(["Name", "22.3 size", "23.0 size"], (map(."code-breakdown"."code-size") | flatten | group_by(.name) | map({name: .[0].name, size22: .[0].size, size23: .[1].size})[] | [.name, .size22, .size23])) | @csv' 22.3.dashboard.json 23.0.dashboard.json > analysis.csv +``` + +#### Loading the data from the json files + + +```python +import json +import pandas as pd + +# load data from JSON file +with open('22.3.dashboard.json', 'r') as f: + data22_3 = json.load(f) +with open('23.0.dashboard.json', 'r') as f: + data23_0 = json.load(f) + +# create dataframes from json data +df22_3 = pd.DataFrame(data22_3) +df23_0 = pd.DataFrame(data23_0) + +``` + + +## Key Observations + +The key questions we want to answer using the aforementioned data are: + +1. Is the code area bigger due to more methods being compiled? +2. Is the code area bigger due to more code being generated per method? +3. Is the heap image bigger because we store more objects in it? +4. Is the heap image bigger because we store more metadata? + +### Code Area Size Increase + +We first answer the code area related questions. + +#### Is the code area bigger due to more methods being compiled? + +To answer this question we get the two lists of the compiled methods and compare their sizes: + + +```python +# Get code-size lists from dataframes +code_size_22_3 = df22_3['code-breakdown']['code-size'] +code_size_23_0 = df23_0['code-breakdown']['code-size'] + +print("Compiled methods with 22_3:", len(code_size_22_3)) +print("Compiled methods with 23_0:", len(code_size_23_0)) +``` + + Compiled methods with 22_3: 46298 + Compiled methods with 23_0: 46299 + + +The results indicate that the answer is no. +In both cases the number of compiled methods is the same (off by 1). +As a result the code size increase is not coming from more methods becoming reachable and compiled. + +#### Is the code area bigger due to more code being generated per method? + +To answer this question we calculate the percentage difference between the compiled methods: + + +```python +import matplotlib.pyplot as plt + +# create dataframes from code_size lists +code_df22_3 = pd.DataFrame(code_size_22_3).rename(columns={'size': 'size-22.3'}) +code_df23_0 = pd.DataFrame(code_size_23_0).rename(columns={'size': 'size-23.0'}) + +# merge dataframes +merged_df = pd.merge(code_df22_3, code_df23_0, on='name', how='outer').fillna(0) + +# create column with size increase as percentage skipping entries with 0 size +percentage_increase = lambda row: ( + ((row['size-23.0'] - row['size-22.3']) / row['size-22.3']) * 100 + if row['size-22.3'] > 0 and row['size-23.0'] > 0 + else 0 +) +merged_df['size-increase'] = merged_df.apply(percentage_increase, axis=1) +``` + +We count the number of methods, the number of those that didn't change size, the number of those that their size increased, and the number of those that their size decreased: + + +```python +total_compiled_methods = len(merged_df) +print(f"Total number of compiled methods: {total_compiled_methods}") +zero_increase_count = (merged_df['size-increase'] == 0).sum() +zero_increase_percent = (zero_increase_count / total_compiled_methods) * 100 +print(f"Number of methods that their compiled size remains the same: {zero_increase_count} ({zero_increase_percent:.2f}%)") +positive_increase_count = (merged_df['size-increase'] > 0).sum() +positive_increase_percent = (positive_increase_count / total_compiled_methods) * 100 +print(f"Number of methods that their compiled size increased: {positive_increase_count} ({positive_increase_percent:.2f}%)") +negative_increase_count = (merged_df['size-increase'] < 0).sum() +negative_increase_percent = (negative_increase_count / total_compiled_methods) * 100 +print(f"Number of methods that their compiled size decreased: {negative_increase_count} ({negative_increase_percent:.2f}%)") +``` + + Total number of compiled methods: 48476 + Number of methods that their compiled size remains the same: 13947 (28.77%) + Number of methods that their compiled size increased: 33351 (68.80%) + Number of methods that their compiled size decreased: 1178 (2.43%) + + +The results indicate that 68.8% of the compiled methods are bigger when compiled by 23.0 in comparison to when they are compiled by 22.3. +But how much bigger? +To answer this we print a histogram of the size increase: + + +```python +# plot histogram +plt.hist(merged_df['size-increase'], bins="auto") +plt.xlabel('Percentage difference') +plt.ylabel('Frequency') +plt.title('Histogram of percentage difference in code size (full range)') +plt.show() +``` + + + +![png](/assets/images/posts/mandrel-23-0-image-size-increase/index_9_0.png) + + + +We observe that due to a large number of methods retaining the same size and due to some outliers the histogram is hard to read. +So we remove the methods with no size changes and limit our focus in the range [-5, 25]: + + +```python +# create column with size increase as percentage skipping entries with 0 size +zero_filter = lambda row: row['size-increase'] if row['size-increase'] != 0 else None +merged_df['size-increase'] = merged_df.apply(zero_filter, axis=1) + +# drop rows with None values +merged_df = merged_df.dropna() + +# plot histogram +plt.hist(merged_df['size-increase'], bins="auto") +plt.xlim(-5, 25) +plt.xlabel('Percentage difference') +plt.ylabel('Frequency') +plt.title('Histogram of percentage difference in code size in the range [-5%, 25%] excluding 0%') +# show the plot +plt.show() +``` + + + +![png](/assets/images/posts/mandrel-23-0-image-size-increase/index_11_0.png) + + + +This plot shows that the majority of the affected methods get a code size increase between 0 and 10 %, which is inline with the overall size increase we observe in the code area. + +##### Why? + +To see why the same methods get compiled to larger machine code when using Mandrel 23.0 we first inspected how many methods are getting inlined in each case. +To do so, we build the native executables with debug info generation enabled using the `-Dquarkus.native.debug.enabled=true` parameter. +To make sure that inline DIEs are included when building with Mandrel 22.3 we also pass the `-Dquarkus.native.additional-build-args=-H:-OmitInlinedMethodDebugLineInfo` option, e.g.: + +```shell +mvn clean package -Pnative -Dquarkus.version=3.2.6.Final\ + -Dquarkus.native.builder-image=quay.io/quarkus/ubi-quarkus-mandrel-builder-image:jdk-17\ + -Dquarkus.native.debug.enabled=true\ + -Dquarkus.native.additional-build-args=-H:-OmitInlinedMethodDebugLineInfo +``` + +After the image is built we count the number of _inlined_ Debug Info Entries (DIEs) using the following command: + +```shell +readelf --debug-dump=info quarkus-runner | grep -i "DW_TAG_inlined_subroutine" | wc -l +``` + +The results are shown in the table below: + +| Mandrel version | 22.3 | 23.0 | Increase % | +| --- | --- | --- | --- | +| Inlined methods | 2798414 | 2817686 | 0.69 % | + +While they indicate a slight increase in the number of inlined methods between Mandrel 22.3 and 23.0, the increase is so small that it doesn't align with the overall code size increase. + +As a next step, we hand-picked a number of methods with different code sizes in the generated native executables and inspected their disassembled code (using `gdb`). + +For example inspecting `InstrumentedUnpooledUnsafeNoCleanerDirectByteBuf.allocateDirect(int)` from `io.netty.buffer.UnpooledByteBufAllocator` using: + +```gdb +(gdb) x/20i 'io.netty.buffer.UnpooledByteBufAllocator$InstrumentedUnpooledUnsafeNoCleanerDirectByteBuf::allocateDirect(int)' +``` + +we see that the one extra byte comes from an additional `nop` between two calls. + +**Mandrel 22.3** + +```asm +sub $0x18,%rsp +cmp 0x8(%r15),%rsp +jbe 0x96ad8f +mov %rdi,0x8(%rsp) +mov %esi,%edi +mov %esi,0x14(%rsp) +call 0xb7e870 +nop +call 0x756e10 +nop +``` + +**Mandrel 23.0** + +```asm +sub $0x18,%rsp +cmp 0x8(%r15),%rsp +jbe 0x96ad8f +mov %rdi,0x8(%rsp) +mov %esi,%edi +mov %esi,0x14(%rsp) +call 0xb7e870 +nop +nop // <==== extra nop +call 0x756e10 +nop +``` + +We observed this pattern in multiple methods which is an indication of some code alignment change. +However, there were also methods with increased compiled code size without having an increased number of `nop`s, hinting that the code size increase is not caused by a single change as we confirm in [Attributing Binary Size Increase to Specific Code Changes](#attributing-binary-size-increase-to-specific-code-changes). + +### Image Heap Size Increase + +Upon initial inspection, it was noted that there was an increase of approximately 650KB in the image heap size. +As a result next we opted to answer whether: + +1. The heap image is bigger because we store more objects in it? +2. The heap image is bigger because we store more metadata? + +#### Is the heap image bigger because we store more objects in it? + +Using the dashboard data we first checked whether the number of objects in the image heap is different between the two versions: + + +```python +# Get heap-size lists from dataframes +heap_size_22_3 = df22_3['heap-breakdown']['heap-size'] +heap_size_23_0 = df23_0['heap-breakdown']['heap-size'] + +# create dataframes from heap_size lists +heap_df22_3 = pd.DataFrame(heap_size_22_3).rename(columns={'size': 'size-22.3', 'count': 'count-22.3'}) +heap_df23_0 = pd.DataFrame(heap_size_23_0).rename(columns={'size': 'size-23.0', 'count': 'count-23.0'}) + +print("Number of objects in image heap with 22.3: ", heap_df22_3['count-22.3'].sum()) +print("Number of objects in image heap with 23.0: ", heap_df23_0['count-23.0'].sum()) +``` + + Number of objects in image heap with 22.3: 340870 + Number of objects in image heap with 23.0: 348063 + + +We observe that the image generated with 22.3 has ~7000 (or roughly 2%) more objects in the image heap. +We then check to see if these additional objects are from different types being instantiated: + + +```python +print("Types in image heap with 22.3:", len(heap_size_22_3)) +print("Types in image heap with 23.0:", len(heap_size_23_0)) +``` + + Types in image heap with 22.3: 3681 + Types in image heap with 23.0: 3679 + + +The results indicate that the number of types in the image heap remain about the same, hinting that 23.0 instantiates more objects of the same types in the image heap. +To see which types are those seeing the larger, in terms of heap size, increase in the image heap we run: + + +```python +# merge dataframes +merged_heap_df = pd.merge(heap_df22_3, heap_df23_0, on='name', how='outer').fillna(0) +merged_heap_df['count-diff'] = merged_heap_df['count-23.0'] - merged_heap_df['count-22.3'] +merged_heap_df['size-diff'] = merged_heap_df['size-23.0'] - merged_heap_df['size-22.3'] +# get top 10 types with the biggest difference in occupied heap size +merged_heap_df.sort_values(by=['size-diff'], ascending=False).head(10)[['name', 'count-diff', 'size-diff']] +``` + + + + +
+ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +
namecount-diffsize-diff
1340[B2042.0894704.0
2384Ljava/lang/invoke/DirectMethodHandle;1293.071920.0
2901Ljava/lang/String;2032.065024.0
3684Ljdk/internal/module/ServicesCatalog$ServicePr...1058.042320.0
2582[Ljava/lang/Object;246.036472.0
898[Ljava/lang/String;9.028024.0
1297Ljava/lang/invoke/MethodType;276.015456.0
3238Ljava/util/concurrent/ConcurrentHashMap$Node;274.013152.0
1065Ljava/util/HashMap;146.010512.0
3244[Ljava/lang/Class;261.09680.0
+
+ + + +The results indicate an increase of ~870KB of bytes in byte arrays, which unfortunately is not very informative, especially combined with the size of other types significantly differing between the two versions (possibly due to GraalVM internal code changes which result in different allocation patterns). + +#### Is the heap image bigger because we store more metadata? + +As shown in [Better understanding what is different between the generated native executables](#better-understanding-what-is-different-between-the-generated-native-executables) there appears to be a significant increase in the number of types registered for reflection (645 in Mandrel 22.3 vs. 4317 in Mandrel 23.0). +This, along with the reported (in the `native-image` output) increase of code metadata, initially led us to think that there is some change in Mandrel that results in more types being registered for reflection. +In the end, as discussed in [Attributing Binary Size Increase to Specific Code Changes](#attributing-binary-size-increase-to-specific-code-changes), contrary to our intuition, the increase is not related to the increase in the reported types registered for reflection which is due to [a fix in the way the reported types registered for reflection are measured](https://github.com/oracle/graal/commit/23d70b802b2dbc9b7d2324a31141c32b6575083f#diff-54ef73a23b10bd907d5869cc88b651fae7fef0467ccfaf8f50472cdd1e114eceR385). +Instead, the increase of the code metadata stored in the image heap is due to [skipping constant folding of reflection methods with side effects](https://github.com/oracle/graal/pull/5156), a fix introduced in 23.0 to prevent undesired effects when folding invocations using reflection. + +## Attributing Binary Size Increase to Specific Code Changes + +At this point, we have a rough understanding of what is different in the generated native executables, but we still don't know why. +Is this increase in the size of native executables well justified? + +To answer this question, we decided to detect the code base changes that resulted in the observed behaviors. +To do so, we used `git bisect`, marking the 22.3 release's commit as _good_ and the 23.0 release's commit as _bad_. +For each commit, we built an instance of Mandrel and compiled our test application to see the code and heap area size. +If the sizes matched the ones from 22.3, we marked the commit as _good_ otherwise we marked it as _bad_. +During this process, we noticed that there were commits resulting in binary sizes bigger than the ones generated with 22.3 but smaller than 23.0. +This confirmed our expectation that the binary size increase was the result of more than one change in the code base. +To reduce the `git bisect` cost we noted down the code and heap area sizes for each tested commit hash. +Then using that info, we replayed the bisect process a few times using the output of `git bisect log` and changing which commits we considered _good_, once we identified the first, second, and so forth change contributing to the binary size increase. + +### Identified Causes of Code Size Increase + +The above process led us to the conclusion that the binary size increase is mainly mainly the result of the following three changes: + +- [**Skipping constant folding of reflection methods with side effects**](https://github.com/oracle/graal/pull/5156): A fix introduced in 23.0 to prevent undesired effects when folding invocations using reflection (e.g. triggering build time initialization of classes that should be run time initialized). + This change is responsible for the image heap size increase. + +- [**Reducing the number of stores that are executed by the serial GC write barriers to improve performance by reducing the number of cache misses**](https://github.com/oracle/graal/pull/5330): This change essentially adds two additional instructions, a `cmpb $0x0,0x30(%rcx,%rax,1)` and a `je`, to each inlined instance of the serial GC write barrier. + The aim of this change is to avoid unnecessary stores in the GC write barriers in order to reduce cache line invalidations and improve performance. + According to our measurements, the total impact of this change is ~1MB increase of code area in our test case which inlines the write barrier 90697 times when using Mandrel 22.3 and 93686 times when using Mandrel 23.0. + To measure the number the barrier was inlined we inspect the number of breakpoint locations set in `gdb` when running `b CardTable.java:91`, i.e. when setting a breakpoint in `com.oracle.svm.core.genscavenge.remset.CardTable#setDirty`. + E.g.: + + ```gdb + (gdb) b CardTable.java:91 + Breakpoint 1 at 0x407574: CardTable.java:91. (93686 locations) + ``` + +- [**Enabling code alignment to compensate for the performance penalty of Intel's Jump Conditional Code Erratum**](https://github.com/oracle/graal/commit/4de58f1b3c484213951622c03d74f3435a20c4ef#diff-991a434bbfc9a6af5514e4609380d5fbfe7618585d5b1b3f11fa2a7431ca7ab0L1388-R1388): According to [Intel's white paper about "Mitigations for Jump Conditional Code Erratum"](https://www.intel.com/content/dam/support/us/en/documents/processors/mitigations-jump-conditional-code-erratum.pdf): + + > Software can compensate for the performance effects of the workaround for this erratum with optimizations that align the code such that jump instructions (and macro-fused jump instructions) do not cross 32-byte boundaries or end on a 32-byte boundary. + > Such aligning can reduce or eliminate the performance penalty caused by the transition of execution from Decoded ICache to the legacy decode pipeline. + + As a result, this change results in an increased number of `nop` instructions in the generated code, but can result in up to 4% performance improvements according to the same document: + + > Intel has observed performance effects associated with the workaround ranging from +0-4% on many industry-standard benchmarks. + > In subcomponents of these benchmarks, Intel has observed outliers higher than the 0-4% range. + > Other workloads not observed by Intel may behave differently. + > Intel has in turn developed software-based tools to minimize the impact on potentially affected applications and workloads. + + According to our measurements, the total impact of this change is ~900KB in our test case. + +## Conclusion + +In conclusion, the increase in the size of native executables produced with Mandrel 23.0 compared to Mandrel 22.3 can be attributed to the above three specific changes in the Mandrel code base. + +While these changes do contribute to the increase in native executable size, they also come with performance and correctness benefits. +Therefore, the larger executables are a trade-off to ensure better application performance and avoid undesired side effects. diff --git a/assets/examples/posts/mandrel-23-0-image-size-increase/quarkus-size-22-3-23-0.ipynb b/assets/examples/posts/mandrel-23-0-image-size-increase/quarkus-size-22-3-23-0.ipynb new file mode 100644 index 0000000000..794ff6d7d1 --- /dev/null +++ b/assets/examples/posts/mandrel-23-0-image-size-increase/quarkus-size-22-3-23-0.ipynb @@ -0,0 +1,741 @@ +{ + "cells": [ + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "---\n", + "title: Exploring why native executables produced with Mandrel 23.0 are bigger than with Mandrel 22.3\n", + "summary: >\n", + " Starting with Quarkus 3.2, the default Mandrel version was updated from 22.3 to 23.0.\n", + " This update introduced various improvements but also resulted in larger native executables.\n", + " To understand the reasons behind this size increase, a comprehensive analysis was conducted, attributing it to specific changes in Mandrel's code base.\n", + "date: \"2023-10-10T13:56:45+03:00\"\n", + "lastmod: \"2023-10-10T17:44:42+03:00\"\n", + "draft: true # Is this a draft? true/false\n", + "toc: true # Show table of contents? true/false\n", + "type: post # Do not modify.\n", + "tags: \n", + " - native \n", + " - graalvm\n", + " - mandrel\n", + " - metrics\n", + " - quarkus\n", + "\n", + "# Generate markdown from this file using\n", + "# jupyter nbconvert index.ipynb --to markdown --NbConvertApp.output_files_dir=.\n", + "---\n", + "\n", + "Starting with Quarkus 3.2 the default Mandrel version was updated from 22.3 to 23.0.\n", + "\n", + "This update brought a number of bugfixes as well as new features like:\n", + "\n", + "1. Better support for profiling and debugging using `perf` and `gdb`.\n", + "2. Finer control over the monitoring features included in the native executable.\n", + "3. Support for more JFR events.\n", + "\n", + "However, it also brought an unwanted side effect.\n", + "The native executables produced with Mandrel 23.0 are bigger than the ones produced with Mandrel 22.3.\n", + "To better understand why that happens we perform a thorough analysis to attribute the size increase to specific changes in Mandrel's code base.\n", + "\n", + "## TL;DR\n", + "\n", + "According to our analysis the binary size increase is attributed to three distinct changes, all of which are well justified:\n", + "\n", + "1. [Skipping constant folding of reflection methods with side effects](https://github.com/oracle/graal/pull/5156).\n", + "2. [Reducing the number of stores that are executed by the serial GC write barriers to improve performance by reducing the number of cache misses](https://github.com/oracle/graal/pull/5330).\n", + "3. [Enabling code alignment to compensate for the performance penalty of Intel's jump conditional code erratum](https://github.com/oracle/graal/commit/4de58f1b3c484213951622c03d74f3435a20c4ef#diff-991a434bbfc9a6af5514e4609380d5fbfe7618585d5b1b3f11fa2a7431ca7ab0L1388-R1388).\n", + "\n", + "## Better understanding what is different between the generated native executables\n", + "\n", + "The first step in our analysis is to understand where the binary size increase comes from.\n", + "Usually such an increase is attributed to one of the following:\n", + "\n", + "1. More code being generated, due to more code becoming reachable.\n", + "2. More code being generated, due to more aggressive inlining.\n", + "3. More data being stored in the image heap, due to more objects being reachable.\n", + "4. More data being stored in the image heap, due to more types being registered for reflection thus requiring more code metadata to be stored.\n", + "\n", + "To perform the analysis we use the [Quarkus startstop test](https://github.com/quarkus-qe/quarkus-startstop) (specifically commit `a8bae846881607e376c7c8a96116b6b50ee50b70`) which generates, starts, tests, and stops small Quarkus applications and measures various time-related metrics (e.g. time-to-first-OK-request) and memory usage.\n", + "We get the test with:\n", + "\n", + "```shell\n", + "git clone https://github.com/quarkus-qe/quarkus-startstop\n", + "cd quarkus-startstop\n", + "git checkout a8bae846881607e376c7c8a96116b6b50ee50b70\n", + "```\n", + "\n", + "and build it with:\n", + "\n", + "```shell\n", + "mvn clean package -Pnative -Dquarkus.version=3.2.6.Final\\\n", + " -Dquarkus.native.builder-image=quay.io/quarkus/ubi-quarkus-mandrel-builder-image:jdk-17\n", + "```\n", + "\n", + "changing the builder image tag to `22.3-java17` for building with Mandrel 22.3.\n", + "\n", + "Looking at the build output (generated by Quarkus in `target/my-app-native-image-sources/my-app-build-output-stats.json`) the main differences between the two builds are in the following metrics:\n", + "\n", + "| Mandrel version | 22.3.3.1 | 23.0.1.2 | Increase %\n", + "| --- | --- | --- | --- |\n", + "| Image Heap Size | 28807168 | 29499392 | 2.4\n", + "| Image Code Area | 27680208 | 29625424 | 7\n", + "| Total Image Size | 56826648 | 59467728 | 4.6\n", + "| Classes registered for reflection | 645 | 4317 | 570\n", + "\n", + "Hinting that any of the reasons 1-4 mentioned above is possible.\n", + "\n", + "### Dashboards\n", + "\n", + "GraalVM and Mandrel provide the `-H:+DashboardAll` and `-H:+DashboardJson` flags that can be used to generate dashboards that contain more information about the generated native executable.\n", + "The resulting dashboard contains a number of metrics and looks like this:\n", + "\n", + "```json\n", + "{\n", + " \"points-to\": {\n", + " \"type-flows\": [\n", + " ...\n", + " ]\n", + " },\n", + " \"code-breakdown\": {\n", + " \"code-size\": [\n", + " {\n", + " \"name\": \"io.smallrye.mutiny.CompositeException.getFirstOrFail(Throwable[]) Throwable\",\n", + " \"size\": 575\n", + " },\n", + " ...\n", + " ]\n", + " },\n", + " \"heap-breakdown\": {\n", + " \"heap-size\": [\n", + " {\n", + " \"name\": \"Lio/vertx/core/impl/VerticleManager$$Lambda$bf09d38f5d19578a0d041ffd0a524c1cbe1843df;\",\n", + " \"size\": 24,\n", + " \"count\": 1\n", + " },\n", + " ...\n", + " ]\n", + " }\n", + "}\n", + "```\n", + "\n", + "Using the aforementioned flags we generate dashboards using both Mandrel 22.3 and 23.0 and compare the results.\n", + "\n", + "To generate the dashboards using Mandrel 23.0 we use the following command:\n", + "\n", + "```shell\n", + "mvn package -Pnative \\\n", + " -Dquarkus.version=3.2.6.Final \\\n", + " -Dquarkus.native.builder-image=quay.io/quarkus/ubi-quarkus-mandrel-builder-image:jdk-17 \\\n", + " -Dquarkus.native.additional-build-args=-H:+DashboardAll,-H:+DashboardJson,-H:DashboardDump=path/to/23.0.dashboard.json\n", + "```\n", + "\n", + "Similarly to generate the dashboards using Mandrel 22.3 we use the following command:\n", + "\n", + "```shell\n", + "mvn package -Pnative \\\n", + " -Dquarkus.version=3.2.6.Final \\\n", + " -Dquarkus.native.builder-image=quay.io/quarkus/ubi-quarkus-mandrel-builder-image:22.3-java17 \\\n", + " -Dquarkus.native.additional-build-args=-H:+DashboardAll,-H:+DashboardJson,-H:DashboardDump=path/to/22.3.dashboard.json\n", + "```\n", + "\n", + "Note: Make sure to change `path/to/` to the path where you would like the dashboard json files to be stored, each file is about 370MB big.\n", + "\n", + "### Analyzing and visualizing the data\n", + "\n", + "To process the data from the dashboards we used a Jupyter notebook, like we do in this article.\n", + "To grab the notebook follow [this link](assets/examples/posts/mandrel-23-0-image-size-increase/quarkus-size-22-3-23-0.ipynb).\n", + "\n", + "For those willing to use a spreadsheet instead, a CSV file can be created to facilitate the analysis in a spreadsheet.\n", + "E.g. using `jq`:\n", + "\n", + "```shell\n", + "jq -r -s '([\"Name\", \"22.3 size\", \"23.0 size\"], (map(.\"code-breakdown\".\"code-size\") | flatten | group_by(.name) | map({name: .[0].name, size22: .[0].size, size23: .[1].size})[] | [.name, .size22, .size23])) | @csv' 22.3.dashboard.json 23.0.dashboard.json > analysis.csv\n", + "```\n", + "\n", + "#### Loading the data from the json files" + ] + }, + { + "cell_type": "code", + "execution_count": 1, + "metadata": {}, + "outputs": [], + "source": [ + "import json\n", + "import pandas as pd\n", + "\n", + "# load data from JSON file\n", + "with open('22.3.dashboard.json', 'r') as f:\n", + " data22_3 = json.load(f)\n", + "with open('23.0.dashboard.json', 'r') as f:\n", + " data23_0 = json.load(f)\n", + "\n", + "# create dataframes from json data\n", + "df22_3 = pd.DataFrame(data22_3)\n", + "df23_0 = pd.DataFrame(data23_0)\n" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "\n", + "## Key Observations\n", + "\n", + "The key questions we want to answer using the aforementioned data are:\n", + "\n", + "1. Is the code area bigger due to more methods being compiled?\n", + "2. Is the code area bigger due to more code being generated per method?\n", + "3. Is the heap image bigger because we store more objects in it? \n", + "4. Is the heap image bigger because we store more metadata?\n", + "\n", + "### Code Area Size Increase\n", + "\n", + "We first answer the code area related questions. \n", + "\n", + "#### Is the code area bigger due to more methods being compiled?\n", + "\n", + "To answer this question we get the two lists of the compiled methods and compare their sizes:" + ] + }, + { + "cell_type": "code", + "execution_count": 2, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Compiled methods with 22_3: 46298\n", + "Compiled methods with 23_0: 46299\n" + ] + } + ], + "source": [ + "# Get code-size lists from dataframes\n", + "code_size_22_3 = df22_3['code-breakdown']['code-size']\n", + "code_size_23_0 = df23_0['code-breakdown']['code-size']\n", + "\n", + "print(\"Compiled methods with 22_3:\", len(code_size_22_3))\n", + "print(\"Compiled methods with 23_0:\", len(code_size_23_0))" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The results indicate that the answer is no.\n", + "In both cases the number of compiled methods is the same (off by 1).\n", + "As a result the code size increase is not coming from more methods becoming reachable and compiled.\n", + "\n", + "#### Is the code area bigger due to more code being generated per method?\n", + "\n", + "To answer this question we calculate the percentage difference between the compiled methods:" + ] + }, + { + "cell_type": "code", + "execution_count": 3, + "metadata": {}, + "outputs": [], + "source": [ + "import matplotlib.pyplot as plt\n", + "\n", + "# create dataframes from code_size lists\n", + "code_df22_3 = pd.DataFrame(code_size_22_3).rename(columns={'size': 'size-22.3'})\n", + "code_df23_0 = pd.DataFrame(code_size_23_0).rename(columns={'size': 'size-23.0'})\n", + "\n", + "# merge dataframes\n", + "merged_df = pd.merge(code_df22_3, code_df23_0, on='name', how='outer').fillna(0)\n", + "\n", + "# create column with size increase as percentage skipping entries with 0 size\n", + "percentage_increase = lambda row: (\n", + " ((row['size-23.0'] - row['size-22.3']) / row['size-22.3']) * 100 \n", + " if row['size-22.3'] > 0 and row['size-23.0'] > 0 \n", + " else 0\n", + ")\n", + "merged_df['size-increase'] = merged_df.apply(percentage_increase, axis=1)" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "We count the number of methods, the number of those that didn't change size, the number of those that their size increased, and the number of those that their size decreased:" + ] + }, + { + "cell_type": "code", + "execution_count": 4, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Total number of compiled methods: 48476\n", + "Number of methods that their compiled size remains the same: 13947 (28.77%)\n", + "Number of methods that their compiled size increased: 33351 (68.80%)\n", + "Number of methods that their compiled size decreased: 1178 (2.43%)\n" + ] + } + ], + "source": [ + "total_compiled_methods = len(merged_df)\n", + "print(f\"Total number of compiled methods: {total_compiled_methods}\")\n", + "zero_increase_count = (merged_df['size-increase'] == 0).sum()\n", + "zero_increase_percent = (zero_increase_count / total_compiled_methods) * 100\n", + "print(f\"Number of methods that their compiled size remains the same: {zero_increase_count} ({zero_increase_percent:.2f}%)\")\n", + "positive_increase_count = (merged_df['size-increase'] > 0).sum()\n", + "positive_increase_percent = (positive_increase_count / total_compiled_methods) * 100\n", + "print(f\"Number of methods that their compiled size increased: {positive_increase_count} ({positive_increase_percent:.2f}%)\")\n", + "negative_increase_count = (merged_df['size-increase'] < 0).sum()\n", + "negative_increase_percent = (negative_increase_count / total_compiled_methods) * 100\n", + "print(f\"Number of methods that their compiled size decreased: {negative_increase_count} ({negative_increase_percent:.2f}%)\")" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The results indicate that 68.8% of the compiled methods are bigger when compiled by 23.0 in comparison to when they are compiled by 22.3.\n", + "But how much bigger?\n", + "To answer this we print a histogram of the size increase:" + ] + }, + { + "cell_type": "code", + "execution_count": 5, + "metadata": {}, + "outputs": [ + { + "data": { + "image/png": "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0KFDqtN733333T0v605CQkIwc+ZMuLm53XPgAf46GtipUyccPHgQzZo1U46s3o927drBaDRi4cKFePXVV0v94vL19YWPjw8OHTpUIgBXhoYNG6JBgwb46quvMHHixFKDgqOjIwIDA7Fq1Sp89NFHyi3zhYWF+Pbbb1GnTh3lt946deqEtWvX4tKlS0roLigowMqVK1XLrMj3iqenJ4YOHYpDhw5h/vz55f5cSVhYGLZt24aNGzcqpz5vFxISglmzZiElJaXEqai70ahRIwDAmTNnSl1+cUWf37/99psqQBe/E/hBKfppnNuPoN26dQvffPON5mUGBQVhw4YN+PPPP5XQU1hYiP/85z+qOi37wB9//AE/Pz/NfQMYpKqcPn36wM/PD61atULNmjWRnJyM+fPnw9vbWzm87O/vD+Cv3xYZMmQI9Ho9fH194evri5EjR+LTTz+FlZUVevbsibNnz2Lq1Knw8vLChAkTAPx1Km3ixImIioqCi4sLXnjhBVy4cAHTp09HrVq17ngIvEhISAj+8Y9/4P3330dQUBBOnDiBGTNmoH79+sjPz38wL9D/Z2VlhTlz5mDQoEEICQnBm2++iZycHMydOxeZmZmYNWvWfS3/119/xfDhw/HKK6/g/PnzePfdd1G7dm2MGjUKwF/XYn366acYMmQIrl69ipdffhkeHh64fPkyDh06hMuXL5c4Ulfcs88+i9DQUHzwwQe4dOkSQkJCYDAYcPDgQTg4OGDs2LGoV68eZsyYgXfffRd//PEHevToARcXF1y6dAl79+5V/hd9L5ycnODt7Y3//ve/6NKlC1xdXeHu7q7cirx06VI0atQIzZo1w/79+zF37twS/8sODw/HV199hZ49e2LGjBnw9PTEd999p/zPvWgfut/XqWg9vXv3xgcffABPT08sX7683CMEWoWHh+OHH37Ac889hwkTJqBZs2YoLCzEuXPnsHnzZkRERCAwMLDcZXzyySfo0KEDnn32Wfztb39DvXr1cO3aNZw+fRo//vgjtm7dek99qlGjBj7++GMMHz4cXbt2xYgRI+Dp6YnTp0/j0KFDWLBgAQBg0aJF6NmzJ7p3746hQ4eidu3auHr1Ko4dO4YDBw6U+MJ50D777DP06dMHbdq0wYQJE1C3bl2cO3cOmzZtwvLlywEAUVFR6NatGzp16oTIyEjY2tri888/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EYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRpUapH7++Wf06dMHZrMZOp0Oa9asUabl5eVh8uTJ8Pf3h6OjI8xmMwYPHoyLFy+qlpGTk4OxY8fC3d0djo6O6Nu3Ly5cuKCqycjIQGhoKIxGI4xGI0JDQ5GZmamqOXfuHPr06QNHR0e4u7tj3LhxyM3NfVBDJyIiokdApQapGzduoHnz5liwYEGJaTdv3sSBAwcwdepUHDhwAKtWrcLJkyfRt29fVV14eDhWr16NmJgY7Ny5E9evX0dISAgKCgqUmoEDByIxMRGxsbGIjY1FYmIiQkNDlekFBQXo3bs3bty4gZ07dyImJgY//PADIiIiHtzgiYiIqPqTKgKArF69utyavXv3CgBJTk4WEZHMzEzR6/USExOj1KSkpIiVlZXExsaKiMjRo0cFgOzevVupSUhIEABy/PhxERHZsGGDWFlZSUpKilKzYsUKMRgMYrFY7noMFotFANzTPFWF9+R1ld0FIiKiSnE/39/V6hopi8UCnU6HJ554AgCwf/9+5OXlITg4WKkxm83w8/PDrl27AAAJCQkwGo0IDAxUatq0aQOj0aiq8fPzg9lsVmq6d++OnJwc7N+//yGMjIiIiKojm8ruwN3Kzs7GO++8g4EDB8LZ2RkAkJaWBltbW7i4uKhqPT09kZaWptR4eHiUWJ6Hh4eqxtPTUzXdxcUFtra2Sk1pcnJykJOTozzPysrSNjgiIiKqlqrFEam8vDy8+uqrKCwsxOeff37HehGBTqdTnt/+7/upKS4qKkq5gN1oNMLLy+uOfSMiIqJHR5UPUnl5eejfvz+SkpIQFxenHI0CAJPJhNzcXGRkZKjmSU9PV44wmUwmXLp0qcRyL1++rKopfuQpIyMDeXl5JY5U3W7KlCmwWCzK4/z585rHSURERNVPlQ5SRSHq1KlT2LJlC9zc3FTTAwICoNfrERcXp7Slpqbi8OHDaNeuHQCgbdu2sFgs2Lt3r1KzZ88eWCwWVc3hw4eRmpqq1GzevBkGgwEBAQFl9s9gMMDZ2Vn1ICIiosdHpV4jdf36dZw+fVp5npSUhMTERLi6usJsNuPll1/GgQMHsG7dOhQUFChHjVxdXWFrawuj0YiwsDBERETAzc0Nrq6uiIyMhL+/P7p27QoAaNy4MXr06IERI0Zg0aJFAICRI0ciJCQEvr6+AIDg4GA0adIEoaGhmDt3Lq5evYrIyEiMGDGC4YiIiIjKVtG3EN6L+Ph4AVDiMWTIEElKSip1GgCJj49XlnHr1i0ZM2aMuLq6ir29vYSEhMi5c+dU67ly5YoMGjRInJycxMnJSQYNGiQZGRmqmuTkZOndu7fY29uLq6urjBkzRrKzs+9pPPz5AyIiournfr6/dSIilZLgHkFZWVkwGo2wWCzV7khWvXfW4+ys3pXdDSIioofufr6/q/Q1UkRERERVGYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaVWqQ+vnnn9GnTx+YzWbodDqsWbNGNV1EMG3aNJjNZtjb26Njx444cuSIqiYnJwdjx46Fu7s7HB0d0bdvX1y4cEFVk5GRgdDQUBiNRhiNRoSGhiIzM1NVc+7cOfTp0weOjo5wd3fHuHHjkJub+yCGTURERI+ISg1SN27cQPPmzbFgwYJSp8+ZMwfR0dFYsGAB9u3bB5PJhG7duuHatWtKTXh4OFavXo2YmBjs3LkT169fR0hICAoKCpSagQMHIjExEbGxsYiNjUViYiJCQ0OV6QUFBejduzdu3LiBnTt3IiYmBj/88AMiIiIe3OCJiIio+pMqAoCsXr1aeV5YWCgmk0lmzZqltGVnZ4vRaJSFCxeKiEhmZqbo9XqJiYlRalJSUsTKykpiY2NFROTo0aMCQHbv3q3UJCQkCAA5fvy4iIhs2LBBrKysJCUlRalZsWKFGAwGsVgsdz0Gi8UiAO5pnqrCe/K6yu4CERFRpbif7+8qe41UUlIS0tLSEBwcrLQZDAYEBQVh165dAID9+/cjLy9PVWM2m+Hn56fUJCQkwGg0IjAwUKlp06YNjEajqsbPzw9ms1mp6d69O3JycrB///4HOk4iIiKqvmwquwNlSUtLAwB4enqq2j09PZGcnKzU2NrawsXFpURN0fxpaWnw8PAosXwPDw9VTfH1uLi4wNbWVqkpTU5ODnJycpTnWVlZdzs8IiIiegRU2SNSRXQ6neq5iJRoK654TWn1WmqKi4qKUi5gNxqN8PLyKrdfRERE9GipskHKZDIBQIkjQunp6crRI5PJhNzcXGRkZJRbc+nSpRLLv3z5sqqm+HoyMjKQl5dX4kjV7aZMmQKLxaI8zp8/f4+jJCIiouqsygap+vXrw2QyIS4uTmnLzc3F9u3b0a5dOwBAQEAA9Hq9qiY1NRWHDx9Watq2bQuLxYK9e/cqNXv27IHFYlHVHD58GKmpqUrN5s2bYTAYEBAQUGYfDQYDnJ2dVQ8iIiJ6fFTqNVLXr1/H6dOnledJSUlITEyEq6sr6tati/DwcMycORM+Pj7w8fHBzJkz4eDggIEDBwIAjEYjwsLCEBERATc3N7i6uiIyMhL+/v7o2rUrAKBx48bo0aMHRowYgUWLFgEARo4ciZCQEPj6+gIAgoOD0aRJE4SGhmLu3Lm4evUqIiMjMWLECIYjIiIiKlsF30F4T+Lj4wVAiceQIUNE5K+fQHj//ffFZDKJwWCQ5557Tn7//XfVMm7duiVjxowRV1dXsbe3l5CQEDl37pyq5sqVKzJo0CBxcnISJycnGTRokGRkZKhqkpOTpXfv3mJvby+urq4yZswYyc7Ovqfx8OcPiIiIqp/7+f7WiYhUYo57pGRlZcFoNMJisVS7I1n13lmPs7N6V3Y3iIiIHrr7+f6ustdIEREREVV1DFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFJEREREGjFIEREREWmkKUglJSVVdD+IiIiIqh1NQeqpp55Cp06d8O233yI7O7ui+0RERERULWgKUocOHUKLFi0QEREBk8mEN998E3v37q3ovhERERFVaZqClJ+fH6Kjo5GSkoIlS5YgLS0NHTp0QNOmTREdHY3Lly9XdD+JiIiIqpz7utjcxsYGL7zwAv79739j9uzZOHPmDCIjI1GnTh0MHjwYqampFdVPIiIioirnvoLUr7/+ilGjRqFWrVqIjo5GZGQkzpw5g61btyIlJQXPP/98RfWTiIiIqMqx0TJTdHQ0lixZghMnTqBXr174+uuv0atXL1hZ/ZXL6tevj0WLFqFRo0YV2lkiIiKiqkRTkPriiy8wbNgwvPHGGzCZTKXW1K1bF4sXL76vzhERERFVZZqC1KlTp+5YY2triyFDhmhZPBEREVG1oOkaqSVLluA///lPifb//Oc/WLZs2X13ioiIiKg60BSkZs2aBXd39xLtHh4emDlz5n13ioiIiKg60BSkkpOTUb9+/RLt3t7eOHfu3H13ioiIiKg60BSkPDw88Ntvv5VoP3ToENzc3O67U0RERETVgaYg9eqrr2LcuHGIj49HQUEBCgoKsHXrVowfPx6vvvpqhXUuPz8ff//731G/fn3Y29vjySefxIwZM1BYWKjUiAimTZsGs9kMe3t7dOzYEUeOHFEtJycnB2PHjoW7uzscHR3Rt29fXLhwQVWTkZGB0NBQGI1GGI1GhIaGIjMzs8LGQkRERI8eTUHqgw8+QGBgILp06QJ7e3vY29sjODgYnTt3rtBrpGbPno2FCxdiwYIFOHbsGObMmYO5c+fi008/VWrmzJmD6OhoLFiwAPv27YPJZEK3bt1w7do1pSY8PByrV69GTEwMdu7cievXryMkJAQFBQVKzcCBA5GYmIjY2FjExsYiMTERoaGhFTYWIiIievToRES0znzy5EkcOnQI9vb28Pf3h7e3d0X2DSEhIfD09FT9HtVLL70EBwcHfPPNNxARmM1mhIeHY/LkyQD+Ovrk6emJ2bNn480334TFYkHNmjXxzTffYMCAAQCAixcvwsvL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EwDQokULHDlyBF988QUGDx6s1BUPcyJyx4BXvKa0+jstZ8qUKZg4caLyPCsri2GKiIjoMXJPQWrIkCGq56+//nqFdqa4WrVqoUmTJqq2xo0b44cffgAAmEwmAH8dUapVq5ZSk56erhylMplMyM3NRUZGhuqoVHp6Otq1a6fUXLp0qcT6L1++XOJo1+0MBoMSKomIiOjxc09BasmSJQ+qH6Vq3749Tpw4oWo7efIkvL29AQD169eHyWRCXFwcWrRoAeCv37javn07Zs+eDQAICAiAXq9HXFwc+vfvDwBITU3F4cOHMWfOHABA27ZtYbFYsHfvXrRu3RoAsGfPHlgsFiVsERERERWn6Qc5H5YJEyagXbt2mDlzJvr374+9e/fiyy+/xJdffgngr9Nx4eHhmDlzJnx8fODj44OZM2fCwcEBAwcOBAAYjUaEhYUhIiICbm5ucHV1RWRkJPz9/dG1a1cAfx3l6tGjB0aMGIFFixYBAEaOHImQkBD4+vpWzuCJiIioyqvSQeqZZ57B6tWrMWXKFMyYMQP169fH/PnzMWjQIKVm0qRJuHXrFkaNGoWMjAwEBgZi8+bNcHJyUmrmzZsHGxsb9O/fH7du3UKXLl2wdOlSWFtbKzXLly/HuHHjlLv7+vbtiwULFjy8wRIREVG1o5MH/RsGj5GsrCwYjUZYLBY4OztXdnfuSb131uPsrN6V3Q0iIqKH7n6+v+/rd6SIiIiIHmcMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaMUgRERERaVStglRUVBR0Oh3Cw8OVNhHBtGnTYDabYW9vj44dO+LIkSOq+XJycjB27Fi4u7vD0dERffv2xYULF1Q1GRkZCA0NhdFohNFoRGho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QK/Xq2pSU1Nx+PBhpaZt27awWCzYu3evUrNnzx5YLBalpjQGgwHOzs6qBxERET0+qvQ1UqNHj8Z3332H//73v3ByclKOPBmNRtjb20On0yE8PBwzZ86Ej48PfHx8MHPmTDg4OGDgwIFKbVhYGCIiIuDm5gZXV1dERkbC398fXbt2BQA0btwYPXr0wIgRI7Bo0SIAwMiRIxESEgJfX9/KGTwRERFVeVU6SH3xxRcAgI4dO6ralyxZgqFDhwIAJk2ahFu3bmHUqFHIyMhAYGAgNm/eDCcnJ6V+3rx5sLGxQf/+/XHr1i106dIFS5cuhbW1tVKzfPlyjBs3Trm7r2/fvliwYMGDHSARERFVazoRkcruxKMiKysLRqMRFoul2p3mq/fOepyd1buyu0FERPTQ3c/3d5W+RooernrvrK/sLhAREVUrDFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFJEREREGjFIEREREWnEIEVERESkEYMUERERkUYMUkREREQaMUgRERERacQgRURERKQRgxQRERGRRgxSRERERBoxSBERERFpxCBFREREpBGDFBEREZFGDFKkUu+d9ZXdBSIiomqDQYqIiIhIIwYpIiIiIo0YpIiIiIg0YpAiIiIi0ohBioiIiEgjBikiIiIijRikqAT+BAIREdHdYZAiIiIi0ohBioiIiEgjBikiIiIijRikiIiIiDRikCIiIiLSiEGKiIiISCMGqWI+//xz1K9fH3Z2dggICMCOHTsqu0tERERURTFI3WblypUIDw/Hu+++i4MHD+LZZ59Fz549ce7cucru2kPH35IiIiK6Mwap20RHRyMsLAzDhw9H48aNMX/+fHh5eeGLL76o7K4RERFRFcQg9f/l5uZi//79CA4OVrUHBwdj165dldSrysWjUkREROWzqewOVBV//vknCgoK4OnpqWr39PREWlpaqfPk5OQgJydHeW6xWAAAWVlZD66jGvm9vwmHp3cvc1pZ6k74j/LvsuYnIiKqzoq+t0XknudlkCpGp9OpnotIibYiUVFRmD59eol2Ly+vB9K3+2WcX7nzExERVWXXrl2D0Wi8p3kYpP4/d3d3WFtblzj6lJ6eXuIoVZEp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", + "text/plain": [ + "
" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "# plot histogram\n", + "plt.hist(merged_df['size-increase'], bins=\"auto\")\n", + "plt.xlabel('Percentage difference')\n", + "plt.ylabel('Frequency')\n", + "plt.title('Histogram of percentage difference in code size (full range)')\n", + "plt.show()" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "We observe that due to a large number of methods retaining the same size and due to some outliers the histogram is hard to read.\n", + "So we remove the methods with no size changes and limit our focus in the range [-5, 25]:" + ] + }, + { + "cell_type": "code", + "execution_count": 6, + "metadata": {}, + "outputs": [ + { + "data": { + "image/png": "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4eHhISpXriztb69evYSJiYlGEpuVlSVq1qypcR8szrWSHw8PD9GlS5dC67ycnL/s77//FjY2NsLPz096+OTeV+fMmaNR9/r168LExESMHTu20G0WlJwDEEeOHNGoW6tWLREQEFDo+oQQwszMLN+koajv+5se69zt9O/f/5WxFnQPKuo1VJznXXHuAfl5neR86NChYunSpWL//v1i8+bNom/fvgKA6Nevn0a9w4cPC7VaLQAIPT096Xny5Zdfilq1ar1xslOYXbt2CT09PY3jL4QQJ06cEKNGjRKbNm0SBw4cEEuXLhXu7u5CX19fREVFadSdMGGCUCgUAoBwcHAQcXFxIj09XdSqVUt8+eWXRY6lKMl5cd7HhIQEYWlpKbp06SJiYmKEnp6e+PzzzzWWK8r97E2S8zd9zgEQ9vb2IjU1VSpLTk4Wenp6IjIyUip79913hZOTk8a58ujRI2FjY/PK5PzmzZsCgMb6cq1du1YAELGxsUKI58/9du3aSXlv48aNxe3bt0VCQoIwNTUVBw4cKHRbRfFa3VpWrlyJo0eP5nnldq8oTKNGjXD69GkMHToUu3btQmpqapG3u3fvXgDI8zVKo0aN4O7uLnUjiI+PR3p6Onr27KlRr0mTJgX+Crh79+75li9atAgNGjSAsbExDAwMYGhoiD/++CPfH3W0b98eenr/d0jd3d0BPP/B04tyy69du1bAngIXL17ErVu3EBwcrLFOc3NzdO/eHfHx8Xjy5EmBy7/Ky32UmzZtChcXF+kYx8bG4v79+wgJCUFWVpb0ysnJQbt27XD06NE8X4O+fAx37twJABg2bFiBcezatQtZWVno37+/xnaMjY3h4+OTp/uIQqFAx44dNcrq1KmDq1evFmm/s7KyMH36dNSqVQtGRkYwMDCAkZEREhISNN7T/fv3w8PDA7Vq1dJY/v3339eYfp3j9KK9e/eidu3aqFu3rkZ5nz59irQ/xbFt2zZ4eHigXr16GrEGBATk21WnVatWsLKykqafPXuGP/74A127doWpqanGOtq3b49nz57lGcGgU6dOGtN16tQBAOn9io2NRWpqKoYOHVrg146XLl3C33//LZ2zL283KSkp364yRfGq+PLzzz//4PLlywgNDYWxsXG+dR4/fowjR46gR48eMDc3l8r19fURHByMGzduSDHv3btX+pr5xXq9evXS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v//+O/z8/FC1alWNclNT0zx1c124cAFRUVGIj49H48aNAQA//vgjvL29cfHiRdSoUeMN94KIiIiI5IAt52Xo9u3b2L59O0JDQ/PMW7NmDWxtbVG7dm1ERETg0aNH0ry4uDioVCopMQeAJk2aQKVSITY2tkxiJyIiIqLSx5bzMrRixQpYWFigW7duGuV9+/ZFlSpVoFarcfbsWYwfPx6nT59GdHQ0ACA5ORl2dnZ51mdnZ4fk5OQCt5eeno709HRpOjU1tYT2hIiIiIhKA5PzMrR06VL07dsXxsbGGuWDBw+W/vbw8ICbmxu8vLxw4sQJNGjQAMDzH5a+TAiRb3muyMhITJkypYSip7eF66fbC5x3ZUaHMoyEiIiIXsZuLWXk4MGDuHjxIj744INX1m3QoAEMDQ2RkJAA4Hm/9du3b+epd/fuXdjb2xe4nvHjxyMlJUV6Xb9+/fV3gIiIiIhKHZPzMrJkyRI0bNgQdevWfWXdc+fOITMzEw4ODgAAb29vpKSk4M8//5TqHDlyBCkpKWjatGmB61EqlbC0tNR4EREREZF8sVvLG0pLS8OlS5ek6cTERJw6dQrW1tZwdnYG8Lyv9y+//II5c+bkWf7y5ctYs2YN2rdvD1tbW5w/fx7h4eGoX78+mjVrBgBwd3dHu3btMHjwYGmIxSFDhiAoKIgjtRARERG9Rdhy/oaOHTuG+vXro379+gCAsLAw1K9fH1988YVUZ/369RBC4P3338+zvJGREf744w8EBASgRo0aGDlyJPz9/RETEwN9fX2p3po1a+Dp6Ql/f3/4+/ujTp06WLVqVenvIBERERGVGYUQQmg7CCobqampUKlUSElJYRcXHfYmPwjlj0mJiMoen9+6hS3nREREREQyweSciIiIiEgmmJwTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCSbnREREREQyweSciIiIiEgmmJwTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCSbnREREREQyweSciIiIiEgmmJwTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCSbnREREREQyweSciIiIiEgmmJwTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCSbnREREREQyweSciIiIiEgmmJwTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJ+Rs6cOAAOnbsCEdHRygUCmzevFlj/oABA6BQKDReTZo00aiTnp6OESNGwNbWFmZmZujUqRNu3LihUefBgwcIDg6GSqWCSqVCcHAwHj58WMp7R0RERERlicn5G3r8+DHq1q2LBQsWFFinXbt2SEpKkl47duzQmD969Ghs2rQJ69evx6FDh5CWloagoCBkZ2dLdfr06YNTp04hKioKUVFROHXqFIKDg0ttv4iIiIio7BloO4DyLjAwEIGBgYXWUSqVUKvV+c5LSUnBkiVLsGrVKrRp0wYAsHr1ajg5OSEmJgYBAQG4cOECoqKiEB8fj8aNGwMAfvzxR3h7e+PixYuoUaNGye4UEREREWkFW87LwL59+2BnZ4fq1atj8ODBuHPnjjTv+PHjyMzMhL+/v1Tm6OgIDw8PxMbGAgDi4uKgUqmkxBwAmjRpApVKJdXJT3p6OlJTUzVeRERERCRfTM5LWWBgINasWYM9e/Zgzpw5OHr0KFq1aoX09HQAQHJyMoyMjGBlZaWxnL29PZKTk6U6dnZ2edZtZ2cn1clPZGSk1EddpVLBycmpBPeMiIiIiEoau7WUsl69ekl/e3h4wMvLCy4uLti+fTu6detW4HJCCCgUCmn6xb8LqvOy8ePHIywsTJpOTU1lgk5EREQkY2w5L2MODg5wcXFBQkICAECtViMjIwMPHjzQqHfnzh3Y29tLdW7fvp1nXXfv3pXq5EepVMLS0lLjRURERETyxeS8jN27dw/Xr1+Hg4MDAKBhw4YwNDREdHS0VCcpKQlnz55F06ZNAQDe3t5ISUnBn3/+KdU5cuQIUlJSpDpEREREVP6xW8sbSktLw6VLl6TpxMREnDp1CtbW1rC2tsbkyZPRvXt3ODg44MqVK/jss89ga2uLrl27AgBUKhVCQ0MRHh4OGxsbWFtbIyIiAp6entLoLe7u7mjXrh0GDx6MxYsXAwCGDBmCoKAgjtRCRERE9BZhcv6Gjh07Bj8/P2k6t493SEgIvv/+e5w5cwYrV67Ew4cP4eDgAD8/P2zYsAEWFhbSMt988w0MDAzQs2dPPH36FK1bt8by5cuhr68v1VmzZg1GjhwpjerSqVOnQsdWJyIiIqLyRyGEENoOgspGamoqVCoVUlJS2P9ch7l+ur3AeVdmdCi1ZYmI6PXw+a1b2OeciIiIiEgmmJwTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZIJDKRJpGUdAISIiolxMzonKQGEJ+NuwPSIiIioZ7NZCRERERCQTTM6JiIiIiGSCyTkRERERkUwwOSciIiIikgkm50REREREMsHknIiIiIhIJjiUIhGVCI7XTkRE9ObYck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCSbnREREREQyweSciIiIiEgmmJwTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCSbnREREREQyweSciIiIiEgmmJwTEREREcmEgbYDIKKCuX66vcB5V2Z0KMNIiIiIqCyw5ZyIiIiISCaYnBMRERERyQSTcyIiIiIimWByTkREREQkE0zOiYiIiIhkgsk5EREREZFMMDknIiIiIpIJJudv6MCBA+jYsSMcHR2hUCiwefNmaV5mZibGjRsHT09PmJmZwdHREf3798etW7c01uHr6wuFQqHx6t27t0adBw8eIDg4GCqVCiqVCsHBwXj48GEZ7CERERERlRX+E6I39PjxY9StWxcDBw5E9+7dNeY9efIEJ06cwMSJE1G3bl08ePAAo0ePRqdOnXDs2DGNuoMHD8bUqVOlaRMTE435ffr0wY0bNxAVFQUAGDJkCIKDg7F169ZS2jPSRYX906PSWi//mRIREdH/YXL+hgIDAxEYGJjvPJVKhejoaI2y+fPno1GjRrh27RqcnZ2lclNTU6jV6nzXc+HCBURFRSE+Ph6NGzcGAPz444/w9vbGxYsXUaNGjRLaGyIiIiLSJnZrKWMpKSlQKBSoUKGCRvmaNWtga2uL2rVrIyIiAo8ePZLmxcXFQaVSSYk5ADRp0gQqlQqxsbEFbis9PR2pqakaLyIiIiKSL7acl6Fnz57h008/RZ8+fWBpaSmV9+3bF1WqVIFarcbZs2cxfvx4nD59Wmp1T05Ohp2dXZ712dnZITk5ucDtRUZGYsqUKSW/I0RERERUKpicl5HMzEz07t0bOTk5WLhwoca8wYMHS397eHjAzc0NXl5eOHHiBBo0aAAAUCgUedYphMi3PNf48eMRFhYmTaempsLJyelNd4WIiIiISgmT8zKQmZmJnj17IjExEXv27NFoNc9PgwYNYGhoiISEBDRo0ABqtRq3b9/OU+/u3buwt7cvcD1KpRJKpfKN4yd5Kq0fbxIREZH2sM95KctNzBMSEhATEwMbG5tXLnPu3DlkZmbCwcEBAODt7Y2UlBT8+eefUp0jR44gJSUFTZs2LbXYiYiIiKhsseX8DaWlpeHSpUvSdGJiIk6dOgVra2s4OjqiR48eOHHiBLZt24bs7Gypj7i1tTWMjIxw+fJlrFmzBu3bt4etrS3Onz+P8PBw1K9fH82aNQMAuLu7o127dhg8eDAWL14M4PlQikFBQRyphYiIiOgtwuT8DR07dgx+fn7SdG4f75CQEEyePBlbtmwBANSrV09jub1798LX1xdGRkb4448/8O233yItLQ1OTk7o0KEDJk2aBH19fan+mjVrMHLkSPj7+wMAOnXqhAULFpTy3hERERFRWWJy/oZ8fX0hhChwfmHzAMDJyQn79+9/5Xasra2xevXqYsdHREREROUH+5wTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCSbnREREREQyweSciIiIiEgmmJwTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCSbnREREREQyweSciIiIiEgmmJwTEREREcmEzibniYmJ2g6BiIiIiEiDzibn77zzDvz8/LB69Wo8e/ZM2+EQEREREcFA2wFoy+nTp7F06VKEh4dj+PDh6NWrF0JDQ9GoUSNth0ZE/5/rp9sLnHdlRocyjISIiKhs6GzLuYeHB+bOnYubN29i2bJlSE5ORvPmzVG7dm3MnTsXd+/e1XaIRERERKRjdDY5z2VgYICuXbvi559/xsyZM3H58mVERESgcuXK6N+/P5KSkrQdIhERERHpCJ1Pzo8dO4ahQ4fCwcEBc+fORUREBC5fvow9e/bg5s2b6Ny5s7ZDJCIiIiIdobN9zufOnYtly5bh4sWLaN++PVauXIn27dtDT+/555UqVapg8eLFqFmzppYjJSIiIiJdobPJ+ffff49BgwZh4MCBUKvV+dZxdnbGkiVLyjgyIiIiItJVOpucJyQkvLKOkZERQkJCyiAaIiIi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k3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCSbnREREREQyweSciIiIiEgmmJwTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCSbnREREREQyweSciIiIiEgmmJwTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCSbnpWzy5MlQKBQaL7VaLc0XQmDy5MlwdHSEiYkJfH19ce7cOY11pKenY8SIEbC1tYWZmRk6deqEGzdulPWuEBEREVEpY3JeBmrXro2kpCTpdebMGWnerFmzMHfuXCxYsABHjx6FWq1G27Zt8ejRI6nO6NGjsWnTJqxfvx6HDh1CWloagoKCkJ2drY3dISIiIqJSYqDtAHSBgYGBRmt5LiEE5s2bhwkTJqBbt24AgBUrVsDe3h5r167Fhx9+iJSUFCxZsgSrVq1CmzZtAACrV6+Gk5MTYmJiEBAQUKb7QkRERESlhy3nZSAhIQGOjo6oUqUKevfujX///RcAkJiYiOTkZPj7+0t1lUolfHx8EBsbCwA4fvw4MjMzNeo4OjrCw8NDqlOQ9PR0pKamaryIiIiISL6YnJeyxo0bY+XKldi1axd+/PFHJCcno2nTprh37x6Sk5MBAPb29hrL2NvbS/OSk5NhZGQEKyurAusUJDIyEiqVSno5OTmV4J4RERERUUljcl7KAgMD0b17d3h6eqJNmzbYvn07gOfdV3IpFAqNZYQQecpeVpQ648ePR0pKivS6fv36a+4FEREREZUFJudlzMzMDJ6enkhISJD6ob/cAn7nzh2pNV2tViMjIwMPHjwosE5BlEolLC0tNV5EREREJF9MzstYeno6Lly4AAcHB1SpUgVqtRrR0dHS/IyMDOzfvx9NmzYFADRs2BCGhoYadZKSknD27FmpDhERERG9HThaSymLiIhAx44d4ezsjDt37mDatGlITU1FSEgIFAoFRo8ejenTp8PNzQ1ubm6YPn06TE1N0adPHwCASqVCaGgowsPDYWNjA2tra0REREjdZIiIiIjo7cHkvJTduHED77//Pv777z9UrFgRTZo0QXx8PFxcXAAAY8eOxdOnTzF06FA8ePAAjRs3xu7du2FhYSGt45tvvoGBgQF69uyJp0+fonXr1li+fDn09fW1tVtEREREVAoUQgih7SCobKSmpkKlUiElJYX9z0uY66fbtR0CkVZdmdFB2yEQvbX4/NYtbDknIqI39qoPqEzeiYiKhj8IJSIiIiKSCSbnREREREQyweSciIiIiEgmmJwTEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTPCfEBERUakr7J8U8R8UERH9H7acExERERHJBJNzIiIiIiKZYHJORERERCQTTM6JiIiIiGSCyTkRERERkUwwOSciIiIikgkm50REREREMsHknIiIiIhIJvhPiIiIqFziPzYiorcRW86JiIiIiGSCyTkRERERkUwwOSciIiIikgkm50REREREMsHknIiIiIhIJpicExERERHJBIdSJCIireKQiERE/4ct50REREREMsHknIiIiIhIJpicExERERHJBJNzIiIiIiKZYHJORERERCQTHK2FiIhkq7CRXIiI3kZsOSciIiIikgkm50REREREMsHknIiIiIhIJpicExERERHJBJNzIiIiIiKZYHJeyiIjI/Huu+/CwsICdnZ26NKlCy5evKhRZ8CAAVAoFBqvJk2aaNRJT0/HiBEjYGtrCzMzM3Tq1Ak3btwoy10hInoruH66vdAXEZE2cSjFUrZ//34MGzYM7777LrKysjBhwgT4+/vj/PnzMDMzk+q1a9cOy5Ytk6aNjIw01jN69Ghs3boV69evh42NDcLDwxEUFITjx49DX1+/zPaHiKg8YJJNROUVk/NSFhUVpTG9bNky2NnZ4fjx42jZsqVUrlQqoVar811HSkoKlixZglWrVqFNmzYAgNWrV8PJyQkxMTEICAgovR0gIiIiojLDbi1lLCUlBQBgbW2tUb5v3z7Y2dmhevXqGDx4MO7cuSPNO378ODIzM+Hv7y+VOTo6wsPDA7GxsQVuKz09HampqRovIiIiIpIvJudlSAiBsLAwNG/eHB4eHlJ5YGAg1qxZgz179mDOnDk4evQoWrVqhfT0dABAcnIyjIyMYGVlpbE+e3t7JCcnF7i9yMhIqFQq6eXk5FQ6O0ZEREREJYLdWsrQ8OHD8ddff+HQoUMa5b169ZL+9vDwgJeXF1xcXLB9+3Z069atwPUJIaBQKAqcP378eISFhUnTqampTNCJiIiIZIwt52VkxIgR2LJlC/bu3YvKlSsXWtfBwQEuLi5ISEgAAKjVamRkZODBgwca9e7cuQN7e/sC16NUKmFpaanxIiIiIiL5YnJeyoQQGD58ODZu3Ig9e/agSpUqr1zm3r17uH79OhwcHAAADRs2hKGhIaKjo6U6SUlJOHv2LJo2bVpqsRMRERFR2WK3llI2bNgwrF27Fr///jssLCykPuIqlQomJiZIS0vD5MmT0b17dzg4OODKlSv47LPPYGtri65du0p1Q0NDER4eDhsbG1hbWyMiIgKenp7S6C1EREREVP4xOS9l33//PQDA19dXo3zZsmUYMGAA9PX1cebMGaxcuRIPHz6Eg4MD/Pz8sGHDBlhYWEj1v/nmGxgYGKBnz554+vQpWrdujeXLl3OMcyKiMlTY+OlXZnQow0iI6G2lEEIIbQdBZSM1NRUqlQopKSnsf17C+A9PiIjJOZUWPr91C/ucExERERHJBJNzIiIiIiKZYHJORERERCQTTM6JiIiIiGSCyTkRERERkUwwOSciIiIikgkm50REREREMsHknIiIiIhIJpicExERERHJhIG2AyAiIiLd9ar/sMz/vEq6hi3nREREREQywZZzIiKiUlZY6zBbhonoRUzOiYiISsCrumcQERUFu7UQEREREckEk3MiIiIiIplgck5EREREJBNMzomIiIiIZILJORERERGRTDA5JyIiIiKSCQ6lSFREHCaNiIiIShuTcyIiIpniPy8i0j3s1kJEREREJBNsOSciItIidpkjohex5ZyIiIiISCbYck5ERPQWYn91ovKJLedERERERDLBlnN6K7HFiIjo9fD+SaRdTM5Jq/gQICIqP3jPJip9TM5J53BkBCJ6G/BeRvR2YnJORESkY5jYE8kXk3MiIiJ6Y+zyQlQyOFoLEREREZFMsOWcyi1+LUtEVD7wfk1UdEzOiYiIqFx63aS/PHWzcf10O3LSn2g7DCpDTM6JiIiISgD73VNJYHKugzwm7YKe0rTYyxV2YymNGxK/BiUiotLwus8sPpeoLCiEEELbQVDZSE1NhUqlgtPon18rOSciIqKS96oPBDnpT3B9Xk+kpKTA0tKyDCMjbWDLeTmzcOFCzJ49G0lJSahduzbmzZuHFi1aaDssIiIiek1skacXcSjFcmTDhg0YPXo0JkyYgJMnT6JFixYIDAzEtWvXtB0aEREREZUAJuflyNy5cxEaGooP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", + "text/plain": [ + "
" + ] + }, + "metadata": {}, + "output_type": "display_data" + } + ], + "source": [ + "# create column with size increase as percentage skipping entries with 0 size\n", + "zero_filter = lambda row: row['size-increase'] if row['size-increase'] != 0 else None\n", + "merged_df['size-increase'] = merged_df.apply(zero_filter, axis=1)\n", + "\n", + "# drop rows with None values\n", + "merged_df = merged_df.dropna()\n", + "\n", + "# plot histogram\n", + "plt.hist(merged_df['size-increase'], bins=\"auto\")\n", + "plt.xlim(-5, 25)\n", + "plt.xlabel('Percentage difference')\n", + "plt.ylabel('Frequency')\n", + "plt.title('Histogram of percentage difference in code size in the range [-5%, 25%] excluding 0%')\n", + "# show the plot\n", + "plt.show()" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "This plot shows that the majority of the affected methods get a code size increase between 0 and 10 %, which is inline with the overall size increase we observe in the code area.\n", + "\n", + "##### Why?\n", + "\n", + "To see why the same methods get compiled to larger machine code when using Mandrel 23.0 we first inspected how many methods are getting inlined in each case.\n", + "To do so, we build the native executables with debug info generation enabled using the `-Dquarkus.native.debug.enabled=true` parameter.\n", + "To make sure that inline DIEs are included when building with Mandrel 22.3 we also pass the `-Dquarkus.native.additional-build-args=-H:-OmitInlinedMethodDebugLineInfo` option, e.g.:\n", + "\n", + "```shell\n", + "mvn clean package -Pnative -Dquarkus.version=3.2.6.Final\\\n", + " -Dquarkus.native.builder-image=quay.io/quarkus/ubi-quarkus-mandrel-builder-image:jdk-17\\\n", + " -Dquarkus.native.debug.enabled=true\\\n", + " -Dquarkus.native.additional-build-args=-H:-OmitInlinedMethodDebugLineInfo\n", + "```\n", + "\n", + "After the image is built we count the number of _inlined_ Debug Info Entries (DIEs) using the following command:\n", + "\n", + "```shell\n", + "readelf --debug-dump=info quarkus-runner | grep -i \"DW_TAG_inlined_subroutine\" | wc -l\n", + "```\n", + "\n", + "The results are shown in the table below:\n", + "\n", + "| Mandrel version | 22.3 | 23.0 | Increase % |\n", + "| --- | --- | --- | --- |\n", + "| Inlined methods | 2798414 | 2817686 | 0.69 % |\n", + "\n", + "While they indicate a slight increase in the number of inlined methods between Mandrel 22.3 and 23.0, the increase is so small that it doesn't align with the overall code size increase.\n", + "\n", + "As a next step, we hand-picked a number of methods with different code sizes in the generated native executables and inspected their disassembled code (using `gdb`).\n", + "\n", + "For example inspecting `InstrumentedUnpooledUnsafeNoCleanerDirectByteBuf.allocateDirect(int)` from `io.netty.buffer.UnpooledByteBufAllocator` using:\n", + "\n", + "```gdb\n", + "(gdb) x/20i 'io.netty.buffer.UnpooledByteBufAllocator$InstrumentedUnpooledUnsafeNoCleanerDirectByteBuf::allocateDirect(int)'\n", + "```\n", + "\n", + "we see that the one extra byte comes from an additional `nop` between two calls.\n", + "\n", + "**Mandrel 22.3**\n", + "\n", + "```asm\n", + "sub $0x18,%rsp\n", + "cmp 0x8(%r15),%rsp\n", + "jbe 0x96ad8f\n", + "mov %rdi,0x8(%rsp)\n", + "mov %esi,%edi\n", + "mov %esi,0x14(%rsp)\n", + "call 0xb7e870\n", + "nop\n", + "call 0x756e10\n", + "nop\n", + "```\n", + "\n", + "**Mandrel 23.0**\n", + "\n", + "```asm\n", + "sub $0x18,%rsp\n", + "cmp 0x8(%r15),%rsp\n", + "jbe 0x96ad8f\n", + "mov %rdi,0x8(%rsp)\n", + "mov %esi,%edi\n", + "mov %esi,0x14(%rsp)\n", + "call 0xb7e870\n", + "nop\n", + "nop // <==== extra nop\n", + "call 0x756e10\n", + "nop\n", + "```\n", + "\n", + "We observed this pattern in multiple methods which is an indication of some code alignment change.\n", + "However, there were also methods with increased compiled code size without having an increased number of `nop`s, hinting that the code size increase is not caused by a single change as we confirm in [Attributing Binary Size Increase to Specific Code Changes](#attributing-binary-size-increase-to-specific-code-changes).\n", + "\n", + "### Image Heap Size Increase\n", + "\n", + "Upon initial inspection, it was noted that there was an increase of approximately 650KB in the image heap size.\n", + "As a result next we opted to answer whether:\n", + "\n", + "1. The heap image is bigger because we store more objects in it? \n", + "2. The heap image is bigger because we store more metadata?\n", + "\n", + "#### Is the heap image bigger because we store more objects in it?\n", + "\n", + "Using the dashboard data we first checked whether the number of objects in the image heap is different between the two versions:" + ] + }, + { + "cell_type": "code", + "execution_count": 7, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Number of objects in image heap with 22.3: 340870\n", + "Number of objects in image heap with 23.0: 348063\n" + ] + } + ], + "source": [ + "# Get heap-size lists from dataframes\n", + "heap_size_22_3 = df22_3['heap-breakdown']['heap-size']\n", + "heap_size_23_0 = df23_0['heap-breakdown']['heap-size']\n", + "\n", + "# create dataframes from heap_size lists\n", + "heap_df22_3 = pd.DataFrame(heap_size_22_3).rename(columns={'size': 'size-22.3', 'count': 'count-22.3'})\n", + "heap_df23_0 = pd.DataFrame(heap_size_23_0).rename(columns={'size': 'size-23.0', 'count': 'count-23.0'})\n", + "\n", + "print(\"Number of objects in image heap with 22.3: \", heap_df22_3['count-22.3'].sum())\n", + "print(\"Number of objects in image heap with 23.0: \", heap_df23_0['count-23.0'].sum())" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "We observe that the image generated with 22.3 has ~7000 (or roughly 2%) more objects in the image heap.\n", + "We then check to see if these additional objects are from different types being instantiated:" + ] + }, + { + "cell_type": "code", + "execution_count": 8, + "metadata": {}, + "outputs": [ + { + "name": "stdout", + "output_type": "stream", + "text": [ + "Types in image heap with 22.3: 3681\n", + "Types in image heap with 23.0: 3679\n" + ] + } + ], + "source": [ + "print(\"Types in image heap with 22.3:\", len(heap_size_22_3))\n", + "print(\"Types in image heap with 23.0:\", len(heap_size_23_0))" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The results indicate that the number of types in the image heap remain about the same, hinting that 23.0 instantiates more objects of the same types in the image heap.\n", + "To see which types are those seeing the larger, in terms of heap size, increase in the image heap we run:" + ] + }, + { + "cell_type": "code", + "execution_count": 9, + "metadata": {}, + "outputs": [ + { + "data": { + "text/html": [ + "
\n", + "\n", + "\n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + " \n", + "
namecount-diffsize-diff
1340[B2042.0894704.0
2384Ljava/lang/invoke/DirectMethodHandle;1293.071920.0
2901Ljava/lang/String;2032.065024.0
3684Ljdk/internal/module/ServicesCatalog$ServicePr...1058.042320.0
2582[Ljava/lang/Object;246.036472.0
898[Ljava/lang/String;9.028024.0
1297Ljava/lang/invoke/MethodType;276.015456.0
3238Ljava/util/concurrent/ConcurrentHashMap$Node;274.013152.0
1065Ljava/util/HashMap;146.010512.0
3244[Ljava/lang/Class;261.09680.0
\n", + "
" + ], + "text/plain": [ + " name count-diff size-diff\n", + "1340 [B 2042.0 894704.0\n", + "2384 Ljava/lang/invoke/DirectMethodHandle; 1293.0 71920.0\n", + "2901 Ljava/lang/String; 2032.0 65024.0\n", + "3684 Ljdk/internal/module/ServicesCatalog$ServicePr... 1058.0 42320.0\n", + "2582 [Ljava/lang/Object; 246.0 36472.0\n", + "898 [Ljava/lang/String; 9.0 28024.0\n", + "1297 Ljava/lang/invoke/MethodType; 276.0 15456.0\n", + "3238 Ljava/util/concurrent/ConcurrentHashMap$Node; 274.0 13152.0\n", + "1065 Ljava/util/HashMap; 146.0 10512.0\n", + "3244 [Ljava/lang/Class; 261.0 9680.0" + ] + }, + "execution_count": 9, + "metadata": {}, + "output_type": "execute_result" + } + ], + "source": [ + "# merge dataframes\n", + "merged_heap_df = pd.merge(heap_df22_3, heap_df23_0, on='name', how='outer').fillna(0)\n", + "merged_heap_df['count-diff'] = merged_heap_df['count-23.0'] - merged_heap_df['count-22.3']\n", + "merged_heap_df['size-diff'] = merged_heap_df['size-23.0'] - merged_heap_df['size-22.3']\n", + "# get top 10 types with the biggest difference in occupied heap size\n", + "merged_heap_df.sort_values(by=['size-diff'], ascending=False).head(10)[['name', 'count-diff', 'size-diff']]" + ] + }, + { + "cell_type": "markdown", + "metadata": {}, + "source": [ + "The results indicate an increase of ~870KB of bytes in byte arrays, which unfortunately is not very informative, especially combined with the size of other types significantly differing between the two versions (possibly due to GraalVM internal code changes which result in different allocation patterns).\n", + "\n", + "#### Is the heap image bigger because we store more metadata?\n", + "\n", + "As shown in [Better understanding what is different between the generated native executables](#better-understanding-what-is-different-between-the-generated-native-executables) there appears to be a significant increase in the number of types registered for reflection (645 in Mandrel 22.3 vs. 4317 in Mandrel 23.0).\n", + "This, along with the reported (in the `native-image` output) increase of code metadata, initially led us to think that there is some change in Mandrel that results in more types being registered for reflection.\n", + "In the end, as discussed in [Attributing Binary Size Increase to Specific Code Changes](#attributing-binary-size-increase-to-specific-code-changes), contrary to our intuition, the increase is not related to the increase in the reported types registered for reflection which is due to [a fix in the way the reported types registered for reflection are measured](https://github.com/oracle/graal/commit/23d70b802b2dbc9b7d2324a31141c32b6575083f#diff-54ef73a23b10bd907d5869cc88b651fae7fef0467ccfaf8f50472cdd1e114eceR385).\n", + "Instead, the increase of the code metadata stored in the image heap is due to [skipping constant folding of reflection methods with side effects](https://github.com/oracle/graal/pull/5156), a fix introduced in 23.0 to prevent undesired effects when folding invocations using reflection.\n", + "\n", + "## Attributing Binary Size Increase to Specific Code Changes\n", + "\n", + "At this point, we have a rough understanding of what is different in the generated native executables, but we still don't know why.\n", + "Is this increase in the size of native executables well justified?\n", + "\n", + "To answer this question, we decided to detect the code base changes that resulted in the observed behaviors.\n", + "To do so, we used `git bisect`, marking the 22.3 release's commit as _good_ and the 23.0 release's commit as _bad_.\n", + "For each commit, we built an instance of Mandrel and compiled our test application to see the code and heap area size.\n", + "If the sizes matched the ones from 22.3, we marked the commit as _good_ otherwise we marked it as _bad_.\n", + "During this process, we noticed that there were commits resulting in binary sizes bigger than the ones generated with 22.3 but smaller than 23.0.\n", + "This confirmed our expectation that the binary size increase was the result of more than one change in the code base.\n", + "To reduce the `git bisect` cost we noted down the code and heap area sizes for each tested commit hash.\n", + "Then using that info, we replayed the bisect process a few times using the output of `git bisect log` and changing which commits we considered _good_, once we identified the first, second, and so forth change contributing to the binary size increase.\n", + "\n", + "### Identified Causes of Code Size Increase\n", + "\n", + "The above process led us to the conclusion that the binary size increase is mainly mainly the result of the following three changes:\n", + "\n", + "- [**Skipping constant folding of reflection methods with side effects**](https://github.com/oracle/graal/pull/5156): A fix introduced in 23.0 to prevent undesired effects when folding invocations using reflection (e.g. triggering build time initialization of classes that should be run time initialized).\n", + " This change is responsible for the image heap size increase.\n", + "\n", + "- [**Reducing the number of stores that are executed by the serial GC write barriers to improve performance by reducing the number of cache misses**](https://github.com/oracle/graal/pull/5330): This change essentially adds two additional instructions, a `cmpb $0x0,0x30(%rcx,%rax,1)` and a `je`, to each inlined instance of the serial GC write barrier.\n", + " The aim of this change is to avoid unnecessary stores in the GC write barriers in order to reduce cache line invalidations and improve performance.\n", + " According to our measurements, the total impact of this change is ~1MB increase of code area in our test case which inlines the write barrier 90697 times when using Mandrel 22.3 and 93686 times when using Mandrel 23.0.\n", + " To measure the number the barrier was inlined we inspect the number of breakpoint locations set in `gdb` when running `b CardTable.java:91`, i.e. when setting a breakpoint in `com.oracle.svm.core.genscavenge.remset.CardTable#setDirty`.\n", + " E.g.:\n", + "\n", + " ```gdb\n", + " (gdb) b CardTable.java:91\n", + " Breakpoint 1 at 0x407574: CardTable.java:91. (93686 locations)\n", + " ```\n", + "\n", + "- [**Enabling code alignment to compensate for the performance penalty of Intel's Jump Conditional Code Erratum**](https://github.com/oracle/graal/commit/4de58f1b3c484213951622c03d74f3435a20c4ef#diff-991a434bbfc9a6af5514e4609380d5fbfe7618585d5b1b3f11fa2a7431ca7ab0L1388-R1388): According to [Intel's white paper about \"Mitigations for Jump Conditional Code Erratum\"](https://www.intel.com/content/dam/support/us/en/documents/processors/mitigations-jump-conditional-code-erratum.pdf):\n", + "\n", + " > Software can compensate for the performance effects of the workaround for this erratum with optimizations that align the code such that jump instructions (and macro-fused jump instructions) do not cross 32-byte boundaries or end on a 32-byte boundary.\n", + " > Such aligning can reduce or eliminate the performance penalty caused by the transition of execution from Decoded ICache to the legacy decode pipeline.\n", + "\n", + " As a result, this change results in an increased number of `nop` instructions in the generated code, but can result in up to 4% performance improvements according to the same document:\n", + " \n", + " > Intel has observed performance effects associated with the workaround ranging from\n", + "0-4% on many industry-standard benchmarks.\n", + " > In subcomponents of these benchmarks, Intel has observed outliers higher than the 0-4% range.\n", + " > Other workloads not observed by Intel may behave differently.\n", + " > Intel has in turn developed software-based tools to minimize the impact on potentially affected applications and workloads.\n", + " \n", + " According to our measurements, the total impact of this change is ~900KB in our test case.\n", + "\n", + "## Conclusion\n", + "\n", + "In conclusion, the increase in the size of native executables produced with Mandrel 23.0 compared to Mandrel 22.3 can be attributed to the above three specific changes in the Mandrel code base.\n", + "\n", + "While these changes do contribute to the increase in native executable size, they also come with performance and correctness benefits.\n", + "Therefore, the larger executables are a trade-off to ensure better application performance and avoid undesired side effects." + ] + } + ], + "metadata": { + "kernelspec": { + "display_name": "Python 3", + "language": "python", + "name": "python3" + }, + "language_info": { + "codemirror_mode": { + "name": "ipython", + "version": 3 + }, + "file_extension": ".py", + "mimetype": "text/x-python", + "name": "python", + "nbconvert_exporter": "python", + "pygments_lexer": "ipython3", + "version": "3.11.6" + } + }, + "nbformat": 4, + "nbformat_minor": 2 +} diff --git a/assets/images/posts/mandrel-23-0-image-size-increase/index_11_0.png b/assets/images/posts/mandrel-23-0-image-size-increase/index_11_0.png new file mode 100644 index 0000000000..84e492d341 Binary files /dev/null and b/assets/images/posts/mandrel-23-0-image-size-increase/index_11_0.png differ diff --git a/assets/images/posts/mandrel-23-0-image-size-increase/index_9_0.png b/assets/images/posts/mandrel-23-0-image-size-increase/index_9_0.png new file mode 100644 index 0000000000..f3d7449757 Binary files /dev/null and b/assets/images/posts/mandrel-23-0-image-size-increase/index_9_0.png differ