Name |
Date |
Size |
#Lines |
LOC |
||
---|---|---|---|---|---|---|
.. | - | - | ||||
data/ | 12-May-2024 | - | 50 | 44 | ||
icd/ | 12-May-2024 | - | 1,741 | 1,413 | ||
layer/ | 12-May-2024 | - | 1,929 | 1,465 | ||
shim/ | 12-May-2024 | - | 1,277 | 948 | ||
CMakeLists.txt | D | 12-May-2024 | 4.7 KiB | 102 | 89 | |
README.md | D | 12-May-2024 | 9.4 KiB | 194 | 162 | |
framework_config.h.in | D | 12-May-2024 | 5.2 KiB | 106 | 92 | |
test_environment.cpp | D | 12-May-2024 | 24.8 KiB | 519 | 455 | |
test_environment.h | D | 12-May-2024 | 16.6 KiB | 397 | 278 | |
test_util.cpp | D | 12-May-2024 | 30 KiB | 723 | 629 | |
test_util.h | D | 12-May-2024 | 41.4 KiB | 982 | 780 |
README.md
1# Loader Testing Framework 2 3The loader testing framework is a mocking environment which fakes many of the global systems that vulkan requires to run. This allows the writing of tests against a known driver, layer, and system configuration. 4 5The framework consists of: 6* Test ICD 7* Test Layer 8* Shim 9* Manifest Writer 10* Utility 11* Test Environment 12* Tests 13 14 15## Running 16 17By default the Vulkan-Loader repo doesn't enable testing. 18 19To turn on building of the tests, set `BUILD_TESTS=ON` in the CMake configuration. 20 21Use the CMake configuration `UPDATE_DEPS=ON` to automatically get all required test dependencies. 22Or Ensure that `googletest` is in the `external` directory. 23And on Windows only, ensure that the `Detours` library is in the `external` directory. 24 25Linux only: The CMake Configuration `TEST_USE_ADDRESS_SANITIZER` can be used to 26enable Address Sanitizer inside the testing framework. 27 28Run the test executables as normal 29 30The executables available are: 31* `test_regression` 32 33Alternatively, in the build directory run `ctest` to start the test framework. 34 35Use the `ctest` command line parameter `--output-on-failure` to printout logs in failing tests 36 37Note: The test framework was not designed to allow multiple tests to be run in parallel due to the extensive use of files and folders on the system. 38 39 40## Components 41 42### Test ICD and Layer 43The Test ICD and Test Layer have much of their configuration available at runtime to allow maximal test setup flexibility. 44However exported functions in the icd or layer library binaries are an integral part of the configuration but are baked into the binary, thus not runtime configurable. 45To account for that there are multiple binaries of the Test ICD and Test Layer, each being a distinct combination of exported functions. 46The `test_icd.cpp` and `test_layer.cpp` files use macro defines to allow the build system to specify which functions should be exported. 47```c 48#if defined(TEST_ICD_EXPORT_NEW_FUNCTION) 49FRAMEWORK_EXPORT VKAPI_ATTR VkResult VKAPI_CALL vkNewFunction() { ... } 50#endif 51``` 52The `CMakeLists.txt` is responsible for creating the various binaries for the Test ICD and Test Layer with the corresponding macros defined to alter which functions are exported. 53Since the test framework needs to know where the binaries are located in the build tree, the `framework_config.h` file contains the absolute paths to all of the libraries that CMake has defined. 54This allows C++ tests to include the header and know exactly what `.dll`/`.so` file to use. 55Since the paths aren't known until build time, CMake processes the `framework_config.h.in` file which contains the names of the libraries that CMake recognizes and replaces the names with the path on the system to the library. 56 57To add a new configuration for a ICD or layer, the changes needed are as follows: 58* Add the relevant functions & macro guards in `test_icd.cpp` or `test_layer.cpp` 59* In the `framework/icd/CMakeLists.txt` or `framework/layer/CMakeLists.txt`, add a new SHARED library using the icd/layer source files. 60 * We are going to use `new_function` as the name of this new configuration for clarity in this documentation. 61```cmake 62add_library(test_icd_export_new_function SHARED ${TEST_ICD_SOURCES}) 63``` 64* Use `target_compile_definitions()` cmake command on the new library to add the specified exports. 65```cmake 66target_compile_definitions(test_icd_export_new_function PRIVATE TEST_ICD_EXPORT_NEW_FUNCTION=1) 67``` 68* Create a new `.def` file with the same name as the library and place it in the `icd\export_definitions` or `layer\export_definitions` folder. 69 * This file is needed in 32 bit windows to prevent name mangling in `.dll`. 70* Add all of the exported functions in this file. 71 * Include all exported functions, not just the newly added functions. 72``` 73LIBRARY test_icd_export_new_function 74EXPORTS 75 <exported_function_0> 76 <exported_function_1> 77 ... 78``` 79* Go back to the `CMakeLists.txt` and add this def file to the library by using `target_sources`, that way the compiler knows which .def file to use for each library 80```cmake 81if (WIN32) 82 target_sources(test_icd_export_new_function PRIVATE export_definitions/test_icd_new_function.def) 83endif() 84``` 85* To make the library accessible to tests, add the name of the library to `framework/framework_config.h.in` 86```c 87#define TEST_ICD_PATH_NEW_FUNCTION "$<TARGET_FILE:test_icd_export_new_function>" 88``` 89 90### Shim 91Because the loader makes many calls to various OS functionality, the framework intercepts certain calls and makes a few of its own calls to OS functionality to allow proper isolation of the loader from the system it is running on. 92The shim is a SHARED library on windows and apple and is a STATIC library on linux. This is due to the nature of the dynamic linker and how overriding functions operates on different operating systems. 93#### Linux 94 95On linux the dynamic linker will use functions defined in the program binary in loaded `.so`'s if the name matches, allowing easy interception of system calls. 96##### Overridden functions 97* opendir 98* access 99* fopen 100 101#### MacOS 102Redirects the following functions: opendir, access fopen. 103The dynamic linker on MacOS requires a bit of tweaking to make it use functions defined in the program binary to override system functions in `.so`'s. 104##### Overridden functions 105* opendir 106* access 107* fopen 108 109#### Windows 110Windows requires a significantly larger number of functions to be intercepted to isolate it sufficiently enough for testing. 111To facilitate that an external library `Detours` is used. 112It is an open source Microsoft library and supports all of the necessary functionality needed for loader testing. 113Note that the loader calls more system functions than are directly overridden with Detours. 114This is due to some functions being used to query other functions, which the shim library intercepts and returns its own versions of system functions instead, such as `CreateDXGIFactory1`. 115 116##### Overridden functions 117* GetSidSubAuthority 118* EnumAdapters2 119* QueryAdapterInfo 120* CM_Get_Device_ID_List_SizeW 121* CM_Get_Device_ID_ListW 122* CM_Locate_DevNodeW 123* CM_Get_DevNode_Status 124* CM_Get_Device_IDW 125* CM_Get_Child 126* CM_Get_DevNode_Registry_PropertyW 127* CM_Get_Sibling 128* GetDesc1 129* CreateDXGIFactory1 130 131### Utility 132 133There are many utilities that the test framework and tests have access to. These include: 134* Including common C and C++ headers 135* `FRAMEWORK_EXPORT` - macro used for exporting shared library funtions 136* Environment Variable helpers: `get_env_var`, `set_env_var`, `remove_env_var` 137* Windows API error handling helpers 138* filesystem abstractions: 139 * `fs::path` - wrapper around std::string that has a similar API to C++17's `filesystem::path` library 140 * `create_folder`/`delete_folder` 141 * `FolderManager` 142 * Creates a new folder with the given name at construction time. 143 * Allows writing manifests and files (eg, icd or layer binaries) 144 * Automatically destroys the folder and all contained files at destruction 145* LibraryWrapper - load and unload `.dll`/`.so`'s automatically 146* DispatchableHandle - helper class for managing the creation and freeing of dispatchable handles 147* VulkanFunctions - Loads the vulkan-loader and queries all used functions from it 148 * If a test needs to use vulkan functions, they must be loaded here 149* Vulkan Create Info Helpers - provide a nice interface for setting up a vulkan creat info struct. 150 * InstanceCreateInfo 151 * DeviceCreateInfo 152 * DeviceQueueCreateInfo 153* Comparison operators for various vulkan structs 154 155### Test Environment 156 157The `test_environment.h/.cpp` contains classes which organize all the disparate parts of the framework into an easy to use entity that allows setting up and configuring the environment tests run in. 158 159The core components are: 160* InstWrapper - helper to construct and then destroy a vulkan instance during tests 161* DeviceWrapper - helper to construct and then destroy a vulkan device during tests 162* PlatformShimWrapper - opens and configures the platform specific shim library 163 * Sets up the overrides 164 * Resets state (clearing out previous test state) 165 * Sets the `VK_LOADER_DEBUG` env var to `all`. 166* TestICD|LayerHandle - corresponds to a single ICD or Layer on the system 167 * Loads the Test ICD/Layer library 168 * Allows easily 'resetting' the Test ICD/Layer 169* TestICD|LayerDetails - holder of data used for constructing ICD and Layers 170 * Contains the name and api version of the icd or layer binary 171 * LayerDetails also contains the manifest, 172* FrameworkEnvironment 173 * Owns the platform shim 174 * Creates folders for the various manifest search locations, including 175 * icd manifest 176 * explicit layer manifests 177 * implicit layer manifests 178 * null - necessary empty folder to make the loader find nothing when we want it to 179 * Sets these folders up as the redirection locations with the platform shim 180 * Allows adding ICD's and Layers 181 * Writes the json manifest file to the correct folder 182 183The `FrameworkEnvironment` class is used to easily create 'environments'. 184 185The `add_XXX()` member functions of `FrameworkEnvironment` make it easy to add drivers and layers to the environment a test runs in. 186 187The `get_test_icd()` and `get_test_layer()` functions allow querying references to the underlying 188drivers and layers that are in the environment, allowing quick modification of their behavior. 189 190The `reset_test_icd()` and `reset_test_layer()` are similar to the above functions but additionally 191reset the layer or driver to its initial state. 192Use this if you need to reset a driver during a test. 193These functions are called on the drivers and layers when the framework is being create in each test. 194