From 9deb428ec768b434f68e33d731a7f0ab340da0fe Mon Sep 17 00:00:00 2001 From: Greg Tatum Date: Wed, 12 Jun 2019 16:21:11 -0500 Subject: [PATCH] Add docs for writing a custom importer --- docs-developer/README.md | 1 + docs-developer/custom-importer.md | 76 +++++++++++++++++++++++++++++++ 2 files changed, 77 insertions(+) create mode 100644 docs-developer/custom-importer.md diff --git a/docs-developer/README.md b/docs-developer/README.md index 623cc89308..62681d8f05 100644 --- a/docs-developer/README.md +++ b/docs-developer/README.md @@ -20,6 +20,7 @@ * [Loading in profiles from various sources](./loading-in-profiles.md) * [Gecko profile format](./gecko-profile-format.md) * [Processed profile format](./processed-profile-format.md) + * [Writing a Custom Profile Importer](./custom-importer) * [Markers](./markers.md) * [Upgrading profiles](./upgrading-profiles.md) * [Potential performance data sources in Gecko](./data-sources.md) diff --git a/docs-developer/custom-importer.md b/docs-developer/custom-importer.md new file mode 100644 index 0000000000..8c41976a16 --- /dev/null +++ b/docs-developer/custom-importer.md @@ -0,0 +1,76 @@ +# Writing a Custom Profile Importer + +The Firefox Profiler supports a few [external profile formats](../src/profile-logic/import) that the profiler imports by converting them to either the [Gecko profile format](./gecko-profile-format.md) or the [processed profile format](./process-profile-format). The good news is that once an importer targets a specific profile version, the Firefox Profiler will always upgrade that profile to the currently supported format. + +## Useful Docs + + * [Gecko profile format docs](./gecko-profile-format.md) + * [processed profile format](./process-profile-format) + +## Useful code links + + * [Gecko profile format type definition](../src/types/gecko-profile.js) + * [processed profile format type definition](../src/types/profile.js) + * [marker payload type definitions](../src/types/markers.js) + * [profiler data structure utilities](../src/profile-logic/data-structures.js) + * [existing importers](../src/profile-logic/import) + +## How to write a profile converter + +The rest of this guide will assume you are targeting the processed profile format. From a high level point of view, the profile is broken down into the basic profile, with meta information, and a list of threads. These threads are grouped in the UI by their PID. The UI also dynamically generates a few "tracks" from other data sources, like the memory track or screenshot track. This guide primarily documents working with threads. + +A good place to start would probably be to view some of the existing [blank profile generating functions](../src/profile-logic/data-structures.js). + +## Some concepts used in the data structures + +The profile format uses a lot of indexes into other data structures. For instances, this is how you would extract out a function name from a profile. (This should work from the web console when viewing a profile on profiler.firefox.com.) + +```js +{ + // Get the first sample out of the first thread: + const threadIndex = 0; + const sampleIndex = 0; + + const thread = profile.threads[threadIndex]; + + // Look up the stack index from the samples table. + const stackIndex = thread.samples.stack[sampleIndex]; + console.log({stackIndex}) + + // Look up the frame from the stack table. + const frameIndex = thread.stackTable.frame[stackIndex]; + console.log({frameIndex}) + + // Look up the function from the frame table. + const funcIndex = thread.frameTable.func[frameIndex]; + console.log({funcIndex}) + + // Look up the string index from the func table. + const stringIndex = thread.funcTable.name[funcIndex]; + console.log({stringIndex}) + + console.log( + 'Function name of the first sample:', + thread.stringTable.getString(stringIndex) + ); +} +``` + +It is probably to read up some on profile format docs for more information on how this structure is used. + +## Tips + + * Use `{ name: 'GeckoMain' }` for the main thread. + * Ensure that the `pid` value is numeric, and points to the proper threads to nest threads in the timeline. + * Processed profiles have their timestamps adjusted so that all processes use the same timeline. + * Gecko profiles adjust the marker and sample start times based on when the process was created (based off of `geckoProfile.meta.startTime`). + +## Samples + +The call tree, flame graph, and stack chart all use the samples data to generate the reports. If your data is shaped like sample data, it can be easy to shove the information in there. At the time of this writing we don't support variable durations. It's assumed 1 samples is equal in length to the sampling interval. This is probably going to change soon. Most likely the profile formats will assume that there is at least one sample in a thread. It might be necessary to fill out the samples with dummy time data if your format isn't needing samples. The sample data should match the time range as the marker data. + +## Markers + +Markers are used in the marker chart, marker table, and for various visualization in the header's timeline. For instance they are drawn as solid squares. Screenshots are encoded as [ScreenshotPayload](https://github.com/firefox-devtools/profiler/search?q=ScreenshotPayload&unscoped_q=ScreenshotPayload) and extracted as a separate track. Markers can be used to encode some tracing types of data, as they can have start and end times. + +One important distinction in the UI is that there is some processing going on everytime you view a profile. The importer should target the RawMarkerTable, while the UI surfaces the `filteredMarkers` on the window object. The filtered markers do some processing to match up start and end markers. This is a bit beyond the scope of this document, but the type definitions should explain things a bit more. It could be a source of confusion when writing and working with markers.