Branch
        Hash :
7181239d
        
        Author :
  
        
        Date :
2025-04-04T14:13:58
        
      
Add long ANGLE traces feature Enables very long Android captures by swapping binary data chunked buffers to/from disk. Bug: b/425728227 Change-Id: I539f72590eece03cfc69d42fc34be9825a9ff1fe Reviewed-on: https://chromium-review.googlesource.com/c/angle/angle/+/6476924 Reviewed-by: Cody Northrop <cnorthrop@google.com> Commit-Queue: Mark Łobodziński <mark@lunarg.com>
ANGLE currently supports a limited OpenGL capture and replay framework.
Limitations:
To build ANGLE with capture and replay enabled update your GN args:
angle_with_capture_by_default = true
Once built with capture enabled by default, ANGLE supports capturing OpenGL ES calls to CPP replay
files. To enable capture, set the ANGLE_CAPTURE_FRAME_START and ANGLE_CAPTURE_FRAME_END
environment variables to define a capture frame range, or use the capture trigger
property on Android. By default the replay will be stored in the current working
directory. The capture files will be named according to the pattern
angle_capture_context{id}_frame{n}.cpp. Each GL Context currently has its own replay sources.
ANGLE will write out data binary blobs for large Texture or Buffer contents to
angle_capture_context{id}_frame{n}.angledata. Replay programs must be able to load data from the
corresponding angledata files.
Some simple environment variables control frame capture:
ANGLE_CAPTURE_ENABLED:    0 to disable capture entirely. Default is 1.      ANGLE_CAPTURE_COMPRESSION:    0 to disable capture compression. Default is 1.      ANGLE_CAPTURE_OUT_DIR=<path>:    ANGLE_CAPTURE_OUT_DIR=samples/capture_replay. Default is the CWD.      ANGLE_CAPTURE_FRAME_START=<n>:    n.      ANGLE_CAPTURE_FRAME_START=2. Default is 0.      ANGLE_CAPTURE_FRAME_END=<n>:    ANGLE_CAPTURE_FRAME_END=4. Default is 0 which disables capture.      ANGLE_CAPTURE_LABEL=<label>:    ANGLE_CAPTURE_LABEL=foo        foo.angledata.gz
foo_context1_0001.cpp
foo_context1_0002.cpp
foo_context1_0003.cpp
foo_context1.cpp
foo_context1.h
foo.json
foo_shared.cpp
...ANGLE_CAPTURE_SERIALIZE_STATE:    1 to enable GL state serialization. Default is 0.      ANGLE_CAPTURE_MAX_RESIDENT_BINARY_SIZE=<n>:    ANGLE_CAPTURE_BLOCK_SIZE=<n>:    A good way to test out the capture is to use environment variables in conjunction with the sample template. For example:
$ ANGLE_CAPTURE_FRAME_END=4 ANGLE_CAPTURE_OUT_DIR=samples/capture_replay out/Debug/simple_texture_2d --use-angle=vulkan
To run a sample replay you can use a template located in
samples/capture_replay. First run your sample and ensure all capture
files are written to samples/capture_replay. You can conveniently use ANGLE_CAPTURE_OUT_DIR.
Make sure ANGLE_CAPTURE_LABEL is left unset during capture to use the default file names.
Then enable the capture_replay_sample via gn args:
angle_build_capture_replay_sample = trueSee samples/BUILD.gn for details. Then build and run your replay sample:
$ autoninja -C out/Debug capture_replay_sample
$ out/Debug/capture_replay_sample
To run your capture on any platform (Windows, Linux, Android, Mac (untested)), you’ll need to
compile it as part of ANGLE’s Trace Replay harness, which is part of angle_perftests.
Create a folder under src/tests/restricted_traces that matches the ANGLE_CAPTURE_LABEL you
used above.
Place all the trace output files into it.  For example, if the label was desktop_test:
src/tests/restricted_traces$ ls -1 desktop_test/
desktop_test.angledata.gz
desktop_test_context1_0001.cpp
desktop_test_context1_0002.cpp
desktop_test_context1_0003.cpp
desktop_test_context1.cpp
desktop_test_context1.h
desktop_test.json
desktop_test_shared.cpp
Then add the label of your trace to
restricted_traces.json.
Note it includes a version with the string.  Just use the number 1 for local changes.
     "dead_trigger_2 1",
+    "desktop_test 1",
     "disney_tsum_tsum 5",Now you should be able to compile and run the perf test including your trace:
autoninja -C out/Debug angle_perftests
ANGLE_CAPTURE_ENABLED=0 out/Debug/angle_perftests --gtest_filter="*desktop_test*" --verboseFor more comprehensive Android capture/replay documentation, see the restricted_traces README file.
In order to capture on Android, the following additional steps must be taken. These steps presume you’ve built and installed the ANGLE APK with capture enabled, and selected ANGLE as the GLES driver for your application.
Create the output directory
Determine your package name:
 export PACKAGE_NAME com.android.gl2jniThen create an output directory that it can write to:
 $ adb shell mkdir -p /sdcard/Android/data/$PACKAGE_NAME/angle_capture
 $ adb shell chmod 777 /sdcard/Android/data/$PACKAGE_NAME/angle_captureSet properties to use for environment variables
On Android, it is difficult to set an environment variable before starting native code. To work around this, ANGLE will read debug system properties before starting each capture and use them to prime environment variables used by the capture code.
As with desktop captures, Android captures can be taken from a starting frame to an ending frame or triggered at an arbitrary frame.
There are other properties that can be set that match 1:1 with the env vars, but they are not required for capture:
 # Optional
 $ adb shell setprop debug.angle.capture.enabled 0
 $ adb shell setprop debug.angle.capture.out_dir foo
 $ adb shell setprop debug.angle.capture.frame_start 0
 $ adb shell setprop debug.angle.capture.label barRun the application, then pull the files to the capture_replay directory
$ cd samples/capture_replay
$ adb pull /sdcard/Android/data/$PACKAGE_NAME/angle_capture replay_files
$ cp replay_files/* .Update your GN args to specifiy which context will be replayed.
By default Context ID 1 will be replayed. On Android, Context ID 2 is more typical, some apps we’ve run go as high as ID 6. Note: this solution is temporary until EGL capture is in place.
 angle_capture_replay_sample_context_id = 2Replay the capture on desktop
Until we have samples building for Android, the replay sample must be run on desktop. We will also be plumbing replay files into perf and correctness tests which will run on Android.
 $ autoninja -C out/Release capture_replay_sample
 $ out/Release/capture_replay_sampleIn some scenarios, you don’t know which frame you want to start on. You’ll only know when target content is being rendered. For that we’ve added a trigger that can allow starting captures at any time.
To use it, set the following environment variable, in addition to all the setup steps above. Set the trigger value equal to the number of frames you’d like to capture.
adb shell setprop debug.angle.capture.trigger 20
When this value is set, ANGLE_CAPTURE_FRAME_START and ANGLE_CAPTURE_FRAME_END will be ignored.
While your content is rendering, wait until you arrive at the scene you’d like to capture. Then set the value back to zero:
adb shell setprop debug.angle.capture.trigger 0ANGLE will detect this change and start recording the requested number of frames, and the trace files will be written to OUT_DIR.
Any number of traces can be captured in succession. After a trace has been captured, reset the TRIGGER to the number of frames to be captured, optionally reset the OUT_DIR location, and again set the trigger back to zero to begin the new trace.
The python script uses the job queue pattern. We spawn n-1 independent worker processes, where n is the value returned by multiprocessing.cpu_count(). Whenever a worker process finishes a job and becomes available, it grabs the next job from a shared job queue and runs that job on its CPU core. When there are no more jobs in the queue, the worker processes terminate and the main process reports results.
   
A job unit is a test batch. Each test has to go through 3 stages: capture run, replay build, and replay run. The test batch batches the replay build stage of multiple tests together, and the replay run stage of multiple tests together.
   
From the command line, navigate to the ANGLE root folder angle then run the command below:
python3 src/tests/capture_replay_tests.py --gtest_filter=*/ES2_Vulkan --keep-temp-files --batch-count=8--gtest_filter to run only specific tests  --keep-temp-files to keep the trace files  --batch-count to set the number of tests in a (capture) batch. More tests in a batch means that
the tests will finish faster, but also means a lower level of granularity.
All command line arguments can be found at the top of the python script.  
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253
# ANGLE OpenGL Frame Capture and Replay
ANGLE currently supports a limited OpenGL capture and replay framework.
Limitations:
 * GLES capture has many unimplemented functions.
 * EGL capture and replay is not yet supported.
 * Mid-execution capture is supported with the Vulkan back-end.
 * Mid-execution capture has many unimplemented features.
 * Capture and replay is currently only tested on desktop platforms.
 * Binary replay is unimplemented. CPP replay is supported.
## Capturing and replaying an application
To build ANGLE with capture and replay enabled update your GN args:
```
angle_with_capture_by_default = true
```
Once built with capture enabled by default, ANGLE supports capturing OpenGL ES calls to CPP replay
files. To enable capture, set the `ANGLE_CAPTURE_FRAME_START` and `ANGLE_CAPTURE_FRAME_END`
environment variables to define a capture frame range, or use the [capture trigger
property][CaptureTrigger] on Android. By default the replay will be stored in the current working
directory. The capture files will be named according to the pattern
`angle_capture_context{id}_frame{n}.cpp`. Each GL Context currently has its own replay sources.
ANGLE will write out data binary blobs for large Texture or Buffer contents to
`angle_capture_context{id}_frame{n}.angledata`. Replay programs must be able to load data from the
corresponding `angledata` files.
## Controlling Frame Capture
Some simple environment variables control frame capture:
 * `ANGLE_CAPTURE_ENABLED`:
   * Set to `0` to disable capture entirely. Default is `1`.
 * `ANGLE_CAPTURE_COMPRESSION`:
   * Set to `0` to disable capture compression. Default is `1`.
 * `ANGLE_CAPTURE_OUT_DIR=<path>`:
   * Can specify an alternate replay output directory. This can either be an
   absolute path, or relative to CWD.
   * Example: `ANGLE_CAPTURE_OUT_DIR=samples/capture_replay`. Default is the CWD.
 * `ANGLE_CAPTURE_FRAME_START=<n>`:
   * Uses mid-execution capture to write "Setup" functions that starts a Context at frame `n`.
   * Example: `ANGLE_CAPTURE_FRAME_START=2`. Default is `0`.
 * `ANGLE_CAPTURE_FRAME_END=<n>`:
   * Example: `ANGLE_CAPTURE_FRAME_END=4`. Default is `0` which disables capture.
 * `ANGLE_CAPTURE_LABEL=<label>`:
   * When specified, files and functions will be labeled uniquely.
   * Example: `ANGLE_CAPTURE_LABEL=foo`
     * Results in filenames like this:
       ```
       foo.angledata.gz
       foo_context1_0001.cpp
       foo_context1_0002.cpp
       foo_context1_0003.cpp
       foo_context1.cpp
       foo_context1.h
       foo.json
       foo_shared.cpp
       ...
       ```
 * `ANGLE_CAPTURE_SERIALIZE_STATE`:
   * Set to `1` to enable GL state serialization. Default is `0`.
 * `ANGLE_CAPTURE_MAX_RESIDENT_BINARY_SIZE=<n>`:
   * Maximum binary data storage space in bytes. Must be a power of 2. Default is 2GB with a useful range of 512MB-4GB.
 * `ANGLE_CAPTURE_BLOCK_SIZE=<n>`:
   * Block size for binary data, in bytes. Must be a power of 2. Default is 256MB, with a useful range of 32-512MB
A good way to test out the capture is to use environment variables in conjunction with the sample
template. For example:
```
$ ANGLE_CAPTURE_FRAME_END=4 ANGLE_CAPTURE_OUT_DIR=samples/capture_replay out/Debug/simple_texture_2d --use-angle=vulkan
```
## Running the capture_replay sample (desktop only)
To run a sample replay you can use a template located in
[samples/capture_replay](../samples/capture_replay). First run your sample and ensure all capture
files are written to `samples/capture_replay`. You can conveniently use `ANGLE_CAPTURE_OUT_DIR`.
Make sure `ANGLE_CAPTURE_LABEL` is left unset during capture to use the default file names.
Then enable the `capture_replay_sample` via `gn args`:
```
angle_build_capture_replay_sample = true
```
See [samples/BUILD.gn](../samples/BUILD.gn) for details. Then build and run your replay sample:
```
$ autoninja -C out/Debug capture_replay_sample
$ out/Debug/capture_replay_sample
```
## Running a perf test replay (all platforms, including Android)
To run your capture on any platform (Windows, Linux, Android, Mac (untested)), you'll need to
compile it as part of ANGLE's Trace Replay harness, which is part of `angle_perftests`.
Create a folder under `src/tests/restricted_traces` that matches the `ANGLE_CAPTURE_LABEL` you
used above.
Place all the trace output files into it.  For example, if the label was `desktop_test`:
```
src/tests/restricted_traces$ ls -1 desktop_test/
desktop_test.angledata.gz
desktop_test_context1_0001.cpp
desktop_test_context1_0002.cpp
desktop_test_context1_0003.cpp
desktop_test_context1.cpp
desktop_test_context1.h
desktop_test.json
desktop_test_shared.cpp
```
Then add the label of your trace to
[restricted_traces.json](../src/tests/restricted_traces/restricted_traces.json).
Note it includes a version with the string.  Just use the number `1` for local changes.
```
     "dead_trigger_2 1",
+    "desktop_test 1",
     "disney_tsum_tsum 5",
```
Now you should be able to compile and run the perf test including your trace:
```
autoninja -C out/Debug angle_perftests
ANGLE_CAPTURE_ENABLED=0 out/Debug/angle_perftests --gtest_filter="*desktop_test*" --verbose
```
## Capturing an Android application
For more comprehensive Android capture/replay documentation, see
the [restricted_traces README file](../src/tests/restricted_traces/README.md).
In order to capture on Android, the following additional steps must be taken. These steps
presume you've built and installed the ANGLE APK with capture enabled, and selected ANGLE
as the GLES driver for your application.
1. Create the output directory
    Determine your package name:
    ```
    export PACKAGE_NAME com.android.gl2jni
    ```
    Then create an output directory that it can write to:
    ```
    $ adb shell mkdir -p /sdcard/Android/data/$PACKAGE_NAME/angle_capture
    $ adb shell chmod 777 /sdcard/Android/data/$PACKAGE_NAME/angle_capture
    ```
2. Set properties to use for environment variables
    On Android, it is difficult to set an environment variable before starting native code.
    To work around this, ANGLE will read debug system properties before starting each capture
    and use them to prime environment variables used by the capture code.
    As with desktop captures, Android captures can be taken from a starting frame to an
    ending frame or triggered at an arbitrary frame.
    There are other properties that can be set that match 1:1 with the env vars, but
    they are not required for capture:
    ```
    # Optional
    $ adb shell setprop debug.angle.capture.enabled 0
    $ adb shell setprop debug.angle.capture.out_dir foo
    $ adb shell setprop debug.angle.capture.frame_start 0
    $ adb shell setprop debug.angle.capture.label bar
    ```
3.  Run the application, then pull the files to the capture_replay directory
    ```
    $ cd samples/capture_replay
    $ adb pull /sdcard/Android/data/$PACKAGE_NAME/angle_capture replay_files
    $ cp replay_files/* .
    ```
4. Update your GN args to specifiy which context will be replayed.
    By default Context ID 1 will be replayed. On Android, Context ID 2 is more typical, some apps
    we've run go as high as ID 6.
    Note: this solution is temporary until EGL capture is in place.
    ```
    angle_capture_replay_sample_context_id = 2
    ```
5. Replay the capture on desktop
    Until we have samples building for Android, the replay sample must be run on desktop.
    We will also be plumbing replay files into perf and correctness tests which will run on Android.
    ```
    $ autoninja -C out/Release capture_replay_sample
    $ out/Release/capture_replay_sample
    ```
### Starting capture at an arbitrary frame
In some scenarios, you don't know which frame you want to start on. You'll only know when target
content is being rendered.  For that we've added a trigger that can allow starting captures at
any time.
To use it, set the following environment variable, in addition to all the setup steps above. Set
the trigger value equal to the number of frames you'd like to capture.
```
adb shell setprop debug.angle.capture.trigger 20
```
When this value is set, `ANGLE_CAPTURE_FRAME_START` and `ANGLE_CAPTURE_FRAME_END` will be ignored.
While your content is rendering, wait until you arrive at the scene you'd like to capture. Then
set the value back to zero:
```
adb shell setprop debug.angle.capture.trigger 0
```
ANGLE will detect this change and start recording the requested number of frames, and the trace
files will be written to OUT_DIR.
Any number of traces can be captured in succession. After a trace has been captured, reset the TRIGGER
to the number of frames to be captured, optionally reset the OUT_DIR location, and again set the
trigger back to zero to begin the new trace.
## Testing
### Regression Testing Architecture
The [python script][link_to_python_script] uses the job queue pattern. We spawn n-1 independent
worker processes, where n is the value returned by multiprocessing.cpu_count(). Whenever a worker
process finishes a job and becomes available, it grabs the next job from a shared job queue and
runs that job on its CPU core. When there are no more jobs in the queue, the worker processes
terminate and the main process reports results.

### Job unit
A job unit is a test batch. Each test has to go through 3 stages: capture run, replay build, and
replay run. The test batch batches the replay build stage of multiple tests together, and the
replay run stage of multiple tests together.

### Running tests
From the command line, navigate to the ANGLE root folder [angle][angle_folder] then run the
command below:
```
python3 src/tests/capture_replay_tests.py --gtest_filter=*/ES2_Vulkan --keep-temp-files --batch-count=8
```
* `--gtest_filter` to run only specific tests
* `--keep-temp-files` to keep the trace files
* `--batch-count` to set the number of tests in a (capture) batch. More tests in a batch means that
the tests will finish faster, but also means a lower level of granularity.
All command line arguments can be found at the top of the [python script][link_to_python_script].
[angle_folder]: ../
[capture_replay_test_folder]: ../src/tests/capture_replay_tests/
[link_to_python_script]: ../src/tests/capture_replay_tests.py
[CaptureTrigger]: ../src/tests/restricted_traces/README.md#trigger-the-capture