[Memory Profiler] Export to csv

Having an export to CSV option available for table view would be a HUGE lifesaver.
We could use the power of spreadsheet applications to really leverage our analysis.

Thank you,
Pieter

Hi,

and thank you for the Feedback, a csv export. As mentioned in your other thread, weā€™re preparing to completely overhaul the table views and underlying database, which are both currently in the unfortunate proto-duction state to be way too involved to maintain, extend and improve upon. And yes, that goes so far as to making it tricky to even add a csv export that would reliable be printing out all the right info.

We are aware that there are a multitude of reasons why one might want to export this data to a csv and have had several requests of that nature already. Weā€™ll try to provide this as soon as it is reasonably possible to slot it into the production. However, weā€™re also always very interested what prompts these requests because it tends to hint at something the Memory Profiler package is currently not delivering on, but could, any maybe even reasonably should provide out of the box (or as something that can be easily extended upon by its users in some form or another). And yes, that includes CI tooling.

So, could you maybe describe what kind of analysis youā€™re expecting to be running over this and/or what kind of Problems you are looking to solve with this, so that we can maybe even deliver that out of the box to you at some point? :slight_smile:

Hi @MartinTilo ,

The usecases I have in mind are mostly:

  • Removing data that Iā€™m currently not interested in (removing entire columns or rows)
  • Quick sorting / grouping / counting / searching data
  • Plotting data
  • Pivot tables for data analysis
  • ā€¦

Some of these points could be implemented in the memory profiler, but I donā€™t think the memory profiler is ever going to (or even should) beat software that is specialized in manipulating a grid of data. Excel is going to be faster to work with and itā€™s 1 less tool to learn if you want somebody else to analyze your data in a straightforward way.

Thank you!

Thanks for the reply and going a bit more into the details here. This is good info and I agree that maybe the Memory Profiler doesnā€™t need to fully replace Excel or Google Sheets for all the range of data investigation one might be able to come up with. Weā€™re instead trying to cover all specific questions or problems related to memory that any user might come across and then make sure that the UI can provide some answers to or insights into that.

Quite honestly, we just need to trim some columns as well that provide marginal or no value. Weā€™ll be looking into reducing this to a more concise form with more details maybe as optional columns.

Iā€™d be more interested in what kind of rows you might find that you want to filter out. Is there any specific problem you are trying to tackle that doesnā€™t require them or where they just add noise?
Is it what you described in the other thread with this:

Once everything is just NativeArrays, performance should become less of an issue but is there some other aspect of quickness there? Is it more about the workflow of how you have to add these kind of filters in the UI, or how you have to re-add them?

I guess this can mean a number of things but is there any workflow/ problem investigation for which you could foresee certain graphs to be useful?

I can understand that some of this might be ā€œIā€™ll know it when I see itā€ and a bit more exploration and experimenting needed to know what youā€™d want out of this. I guess once we do have csv support Iā€™ll also be asking for anyone willing to share what they did with the data to send it our way as feedback or suggestions. Still would be good to have any obvious shortcomings of the current solution mapped out before we rework it :slight_smile:

Indeed itā€™s partially filtering rows based on what Iā€™m not interested in but the filters already do an OK job at that (except what was explained in the thread you previously quoted from), but it really depends on the usecase.

The last few weeks Iā€™m trying to understand which objects have been changed between 2 sessions (old build vs new build).
In that case, IMO the best way to deduce which objects have changed, is by looking at the name of the objects and their size. (sort size high to low, compare names between old and new, figure out which assets are gone or new.)
That means I donā€™t really need the Name, Index, Value, Address, Unique String columns.
But I might need them in other research areas maybe?

Quick here refers to the performance of the tool, not the workflow. Iā€™m personally quite ok with the way the filters are implemented in terms of UI, but in general the tool often feels sluggish when manipulating data or filters. Glad to hear thatā€™s going to be improved with NativeArrays :slight_smile:

As an example:
if we can export to CSV, the captures could be taken by an automated build system which then generates small reports, this way you can track memory usage for specific places in game over the development of your game, and show that in a line graph.

Since you already show a memory map view, I think most usages for plotting would come from plotting values from multiple snapshots easily.

Having CSV support in general would allow the user a bit more freedom in how they work with the data and if something in the memory profiler is a bit bugged, or not as straightforward as weā€™d like, this would allow the user to work in a familiar tool and avoids having to wait for an update to the package, which can take weeks or months depending on the urgency of the request / bug.

Thank you for taking the time to get user feedback!

Resurrecting because this is the first result in Google for exporting Memory Profiler to CSV.

A workaround is to expose the internals of the MemoryProfiler package to custom Editor windows in the project, then get a reference to the MemoryProfilerWindow, and from here extract the data from the loaded snapshot.

See the following snippet that exports the type names and sizes for all managed objects.

To create a local copy of the package, copy it from the Library PackageCache into the Packages folder of the project (not into Assets). There is no need to change the manifest; just copy the folder and the Editor will update itself.

To use IntelliSense and the Debugger, you also need to make sure the Editor will generate .csproj files for the local package. This is set in ā€œPreferences ā†’ External Tools ā†’ Generate .csproj files for:ā€.
(I have Embedded packages, Local packages, Registry packages and Packages from unknown sources set).

using System.Collections;
using System.Collections.Generic;
using UnityEngine;
using UnityEditor;
using System;
using System.Linq;
using Unity.MemoryProfiler.Editor;
using Unity.MemoryProfiler.Editor.UI;
using Unity.MemoryProfiler.Editor.EnumerationUtilities;
using Unity.MemoryProfiler.Editor.Format;
using System.IO;

namespace MySample
{
    /// <summary>
    /// This Window contains additional methods for exploring snapshots loaded by the MemoryProfilerWindow.
    /// </summary>
    public class MemoryProfilerHelperWindow : EditorWindow
    {
        [MenuItem("MySample/Memory Profiler Helper")]
        static void Init()
        {
            MemoryProfilerHelperWindow window = (MemoryProfilerHelperWindow)EditorWindow.GetWindow(typeof(MemoryProfilerHelperWindow));
            window.Show();
        }

        public class QnDCSV
        {
            private TextWriter writer;

            public QnDCSV(string filename)
            {
                writer = new StreamWriter(filename);
            }

            public void Next(string token)
            {
                writer.Write(token);
                writer.Write(",");
            }

            public void NextLine()
            {
                writer.Write("\n");
            }

            public void Close()
            {
                writer.Dispose();
            }
        }

        void OnGUI()
        {
            if (GUILayout.Button("Save"))
            {
                // Get the opened window Memory Profiler Window.
                // We know the type of this from the UI Toolkit Debugger.
                // To access it we need to copy the Package locally, and add the
                // attribute:
                //     [assembly: InternalsVisibleTo("Assembly-CSharp-Editor")]
                // to the namespace (MemoryProfilerWindow.cs line 44).

                MemoryProfilerWindow window = EditorWindow.GetWindow<MemoryProfilerWindow>();

                // Get the current Mode (UIState::BaseMode) from the UI, which
                // will give us access to the tables cached when the Mode is
                // first created.

                var table = (ObjectAllManagedTable)window.UIState.FirstMode.GetSchema().GetTableByName("AllManagedObjects");

                // Get the Columns we are interested in...
                // Look at the ObjectListTable.cs ~860 for the column names (or use Search in Project)

                List<Unity.MemoryProfiler.Editor.Database.Column> columns = new List<Unity.MemoryProfiler.Editor.Database.Column>();
                columns.Add(table.GetColumnByName("Type"));
                columns.Add(table.GetColumnByName("OwnedSize"));
                // columns.Add(table.GetColumnByName("NativeSize")); // Not available in the AllManagedObjects, but in some others.

                QnDCSV csv = new QnDCSV(@"D:\3drepo\ISSUE_434\types.csv");

                // Check all the columns have the same count.

                var expected = table.GetRowCount();
                foreach (var column in columns)
                {
                    Debug.Assert(column.GetRowCount() == expected);
                }

                // Finally write the CSV

                for (int i = 0; i < table.GetRowCount(); i++)
                {
                    foreach (var column in columns)
                    {
                        // The formatter is used to convert the type to a string only. The result is the 'raw' value in string format.
                        // (i.e. the value will always be in bytes for a size column)
                        csv.Next(
                            column.GetRowValueString(i, Unity.MemoryProfiler.Editor.Database.DefaultDataFormatter.Instance));
                    }

                    csv.NextLine();
                }
                csv.Close();
            }
        }
    }
}
2 Likes

Yes, that would be one way of doing it but will require rework if you want to eventually use version 1.0 of the package, where ObjectAllManagedTable and friends have been removed from the codebase.

Any update on this? Seems like really needed feature missing. The current functionality for filtering objects is not sufficient. We need to be able to filter and sort by any field, use regular expressions, etc.

1 Like

CSV export could be interesting. Iā€™m stuck on 2021.3 (itā€™s not easy to transfer all created content to later versions just like that). When comparing 2 snapshots, I see a ā€˜new in B (110)ā€™ and ā€˜deleted in B (101)ā€™ for example. Just knowing which 9 donā€™t match would be so helpful. Right now I canā€™t even export or copy/paste the text to other software to match the 101 deletes vs the 110 news.

No updates on this.

@ruudvangaal : you donā€™t need to update your project or itā€™s contents to use the 1.0 or 1.1 versions of the Memory Profiler. Just installing a 2022.2+ version of the Unity Editor, creating an empty project with that version, installing the memory Profiler in that empty project and importing the snapshots into it allows you to use the new UI on data from older Unity versions.

For your case, Iā€™d recommend that you try version 1.1.0-pre.1 (which might require that you modify the manifest to specify that version). That version might make it more obvious what the 9 extra new objects are.

1 Like

@MartinTilo Thanks for the tip, Iā€™ll give the new Memory Profiler a try!

1 Like

Just installed Unity2022.3.24f1 and Memory Profiler 1.1.0.
Are there any hints on how to correctly view dumps saved from older versions of unity/memory profiler?
Looks like new version of profiler incorrectly reads dumps saved from version Unity2021.3.34+Memory Profiler 0.7.5 - almost all managed objects are absent in the table, comparing to what is displayed in version 0.7.5.

That shouldnā€™t be the case? Are you looking at the Managed Objects part of the tree in the All Off Memory Table? If so, could you please file a bug with some examples of objects that are missing? If anything, 1.1 should be finding more managed objects as previous versions ignored those objects that were only referenced from fields on value types.

The Memory Profiler Package version the snapshot comes from also doesnā€™t matter, only the Unity version of the Player (or the Editor if capturing the Editor).