mirror of
https://github.com/intel/compute-runtime.git
synced 2025-12-21 17:29:14 +08:00
Zebin: set kernel barriers based on ext funcs
This change allows for modifying kernel's barrier count based on called external functions metadata passed via zeInfo section in zebin. Added parsing external functions metadata. Added resolving external functions call graph. Added updating kernel barriers based on called external functions. Added support for L0 dynamic link. Signed-off-by: Krystian Chmielewski <krystian.chmielewski@intel.com>
This commit is contained in:
committed by
Compute-Runtime-Automation
parent
bae9e6f5b5
commit
0ccce5a6d7
@@ -0,0 +1,199 @@
|
||||
/*
|
||||
* Copyright (C) 2022 Intel Corporation
|
||||
*
|
||||
* SPDX-License-Identifier: MIT
|
||||
*
|
||||
*/
|
||||
|
||||
#include "shared/source/compiler_interface/external_functions.h"
|
||||
#include "shared/source/kernel/kernel_descriptor.h"
|
||||
|
||||
#include "gtest/gtest.h"
|
||||
|
||||
#include <vector>
|
||||
|
||||
using namespace NEO;
|
||||
|
||||
TEST(DependencyResolverTests, GivenEmptyGraphReturnEmptyResolve) {
|
||||
std::vector<std::vector<size_t>> graph = {};
|
||||
DependencyResolver resolver(graph);
|
||||
const auto &resolve = resolver.resolveDependencies();
|
||||
EXPECT_TRUE(resolve.empty());
|
||||
}
|
||||
|
||||
TEST(DependencyResolverTests, GivenGraphWithLoopReturnEmptyResolveAndSetLoopDeteckted) {
|
||||
/*
|
||||
0 -> 1
|
||||
^ |
|
||||
| v
|
||||
3 <- 2
|
||||
*/
|
||||
std::vector<std::vector<size_t>> graph = {{1}, {2}, {3}, {0}};
|
||||
DependencyResolver resolver(graph);
|
||||
const auto &resolve = resolver.resolveDependencies();
|
||||
EXPECT_TRUE(resolve.empty());
|
||||
EXPECT_TRUE(resolver.hasLoop());
|
||||
}
|
||||
|
||||
TEST(DependencyResolverTests, GivenOneConnectedGraphReturnCorrectResolve) {
|
||||
/*
|
||||
0 -> 1 -> 2
|
||||
^
|
||||
|
|
||||
3 -> 4
|
||||
*/
|
||||
std::vector<std::vector<size_t>> graph = {{1}, {2}, {}, {1, 4}, {}};
|
||||
std::vector<size_t> expectedResolve = {2, 1, 0, 4, 3};
|
||||
DependencyResolver resolver(graph);
|
||||
const auto &resolve = resolver.resolveDependencies();
|
||||
EXPECT_EQ(expectedResolve, resolve);
|
||||
}
|
||||
|
||||
TEST(DependencyResolverTests, GivenMultipleDisconnectedGraphsReturnCorrectResolve) {
|
||||
/*
|
||||
0
|
||||
1 -> 2 -> 5
|
||||
4 -> 3
|
||||
*/
|
||||
std::vector<std::vector<size_t>> graph = {{}, {2}, {5}, {}, {3}, {}};
|
||||
std::vector<size_t> expectedResolve = {5, 2, 1, 3, 4};
|
||||
DependencyResolver resolver(graph);
|
||||
const auto &resolve = resolver.resolveDependencies();
|
||||
EXPECT_EQ(expectedResolve, resolve);
|
||||
}
|
||||
|
||||
struct ExternalFunctionsTests : public ::testing::Test {
|
||||
void SetUp() override {}
|
||||
void TearDown() override {}
|
||||
|
||||
void addExternalFunction(const std::string &functionName, uint8_t barrierCount) {
|
||||
funcNameToId[functionName] = extFuncInfoStorage.size();
|
||||
extFuncInfoStorage.push_back(ExternalFunctionInfo{functionName, barrierCount, 128U, 8U});
|
||||
}
|
||||
void addKernel(const std::string &kernelName) {
|
||||
kernelDescriptorStorage.push_back(std::make_unique<KernelDescriptor>());
|
||||
auto &kd = *kernelDescriptorStorage.rbegin();
|
||||
kd->kernelMetadata.kernelName = kernelName;
|
||||
nameToKernelDescriptor[kernelName] = kd.get();
|
||||
}
|
||||
void addFuncDependency(const std::string &calleeName, const std::string &callerName) {
|
||||
funcDependenciesStorage.push_back({calleeName, callerName});
|
||||
}
|
||||
void addKernelDependency(const std::string &calleeName, const std::string &kernelCallerName) {
|
||||
kernelDependenciesStorage.push_back({calleeName, kernelCallerName});
|
||||
}
|
||||
|
||||
void clear() {
|
||||
extFuncInfoStorage.clear();
|
||||
kernelDependenciesStorage.clear();
|
||||
funcDependenciesStorage.clear();
|
||||
kernelDescriptorStorage.clear();
|
||||
funcNameToId.clear();
|
||||
nameToKernelDescriptor.clear();
|
||||
}
|
||||
void set() {
|
||||
auto toPtrVec = [](auto &inVec, auto &outPtrVec) {
|
||||
outPtrVec.resize(inVec.size());
|
||||
for (size_t i = 0; i < inVec.size(); i++) {
|
||||
outPtrVec[i] = &inVec[i];
|
||||
}
|
||||
};
|
||||
toPtrVec(extFuncInfoStorage, extFuncInfo);
|
||||
toPtrVec(funcDependenciesStorage, functionDependencies);
|
||||
toPtrVec(kernelDependenciesStorage, kernelDependencies);
|
||||
}
|
||||
|
||||
ExternalFunctionInfosT extFuncInfo;
|
||||
FunctionDependenciesT functionDependencies;
|
||||
KernelDependenciesT kernelDependencies;
|
||||
KernelDescriptorMapT nameToKernelDescriptor;
|
||||
FuncNameToIdMapT funcNameToId;
|
||||
|
||||
protected:
|
||||
std::vector<ExternalFunctionInfo> extFuncInfoStorage;
|
||||
std::vector<ExternalFunctionUsageKernel> kernelDependenciesStorage;
|
||||
std::vector<ExternalFunctionUsageExtFunc> funcDependenciesStorage;
|
||||
std::vector<std::unique_ptr<KernelDescriptor>> kernelDescriptorStorage;
|
||||
};
|
||||
|
||||
TEST_F(ExternalFunctionsTests, GivenMissingExtFuncInLookupMapWhenResolvingExtFuncDependenciesThenReturnError) {
|
||||
addFuncDependency("fun1", "fun0");
|
||||
set();
|
||||
auto error = resolveExtFuncDependencies(extFuncInfo, funcNameToId, functionDependencies);
|
||||
EXPECT_EQ(ERROR_EXTERNAL_FUNCTION_INFO_MISSING, error);
|
||||
clear();
|
||||
|
||||
addFuncDependency("fun1", "fun0");
|
||||
addExternalFunction("fun1", 0);
|
||||
set();
|
||||
error = resolveExtFuncDependencies(extFuncInfo, funcNameToId, functionDependencies);
|
||||
EXPECT_EQ(ERROR_EXTERNAL_FUNCTION_INFO_MISSING, error);
|
||||
clear();
|
||||
|
||||
addFuncDependency("fun1", "fun0");
|
||||
addExternalFunction("fun0", 0);
|
||||
set();
|
||||
error = resolveExtFuncDependencies(extFuncInfo, funcNameToId, functionDependencies);
|
||||
EXPECT_EQ(ERROR_EXTERNAL_FUNCTION_INFO_MISSING, error);
|
||||
}
|
||||
|
||||
TEST_F(ExternalFunctionsTests, GivenLoopWhenResolvingExtFuncDependenciesThenReturnError) {
|
||||
addExternalFunction("fun0", 0);
|
||||
addExternalFunction("fun1", 0);
|
||||
addFuncDependency("fun0", "fun1");
|
||||
addFuncDependency("fun1", "fun0");
|
||||
set();
|
||||
auto error = resolveExtFuncDependencies(extFuncInfo, funcNameToId, functionDependencies);
|
||||
EXPECT_EQ(ERROR_LOOP_DETECKTED, error);
|
||||
}
|
||||
|
||||
TEST_F(ExternalFunctionsTests, GivenMissingExtFuncInLookupMapWhenResolvingKernelDependenciesThenReturnError) {
|
||||
addKernel("kernel");
|
||||
addKernelDependency("fun0", "kernel");
|
||||
set();
|
||||
auto error = resolveKernelDependencies(extFuncInfo, funcNameToId, kernelDependencies, nameToKernelDescriptor);
|
||||
EXPECT_EQ(ERROR_EXTERNAL_FUNCTION_INFO_MISSING, error);
|
||||
}
|
||||
|
||||
TEST_F(ExternalFunctionsTests, GivenMissingKernelInLookupMapWhenResolvingKernelDependenciesThenReturnError) {
|
||||
addExternalFunction("fun0", 0);
|
||||
addKernelDependency("fun0", "kernel");
|
||||
set();
|
||||
auto error = resolveKernelDependencies(extFuncInfo, funcNameToId, kernelDependencies, nameToKernelDescriptor);
|
||||
EXPECT_EQ(ERROR_KERNEL_DESCRIPTOR_MISSING, error);
|
||||
}
|
||||
|
||||
TEST_F(ExternalFunctionsTests, GivenNoDependenciesWhenResolvingBarrierCountThenReturnSuccess) {
|
||||
set();
|
||||
auto error = resolveBarrierCount(extFuncInfo, kernelDependencies, functionDependencies, nameToKernelDescriptor);
|
||||
EXPECT_EQ(RESOLVE_SUCCESS, error);
|
||||
}
|
||||
|
||||
TEST_F(ExternalFunctionsTests, GivenMissingExtFuncInExtFuncDependenciesWhenResolvingBarrierCountThenReturnError) {
|
||||
addFuncDependency("fun0", "fun1");
|
||||
set();
|
||||
auto error = resolveBarrierCount(extFuncInfo, kernelDependencies, functionDependencies, nameToKernelDescriptor);
|
||||
EXPECT_EQ(ERROR_EXTERNAL_FUNCTION_INFO_MISSING, error);
|
||||
}
|
||||
|
||||
TEST_F(ExternalFunctionsTests, GivenMissingExtFuncInKernelDependenciesWhenResolvingBarrierCountThenReturnError) {
|
||||
addKernelDependency("fun0", "kernel");
|
||||
addKernel("kernel");
|
||||
set();
|
||||
auto error = resolveBarrierCount(extFuncInfo, kernelDependencies, functionDependencies, nameToKernelDescriptor);
|
||||
EXPECT_EQ(ERROR_EXTERNAL_FUNCTION_INFO_MISSING, error);
|
||||
}
|
||||
|
||||
TEST_F(ExternalFunctionsTests, GivenValidFunctionAndKernelDependenciesWhenResolvingBarrierCountThenSetAppropriateBarrierCountAndReturnSuccess) {
|
||||
addKernel("kernel");
|
||||
addExternalFunction("fun0", 1U);
|
||||
addExternalFunction("fun1", 2U);
|
||||
addFuncDependency("fun1", "fun0");
|
||||
addKernelDependency("fun0", "kernel");
|
||||
set();
|
||||
auto error = resolveBarrierCount(extFuncInfo, kernelDependencies, functionDependencies, nameToKernelDescriptor);
|
||||
EXPECT_EQ(RESOLVE_SUCCESS, error);
|
||||
EXPECT_EQ(2U, extFuncInfo[funcNameToId["fun0"]]->barrierCount);
|
||||
EXPECT_EQ(2U, extFuncInfo[funcNameToId["fun1"]]->barrierCount);
|
||||
EXPECT_EQ(2U, nameToKernelDescriptor["kernel"]->kernelAttributes.barrierCount);
|
||||
}
|
||||
Reference in New Issue
Block a user