blob: 2a5ef477959b6849d406486bf3140d103e766f9c [file] [log] [blame]
from .common.codegen import CodeGen, VulkanWrapperGenerator, VulkanAPIWrapper
from .common.vulkantypes import \
VulkanAPI, makeVulkanTypeSimple, iterateVulkanType, DISPATCHABLE_HANDLE_TYPES, NON_DISPATCHABLE_HANDLE_TYPES
from .transform import TransformCodegen, genTransformsForVulkanType
from .wrapperdefs import API_PREFIX_MARSHAL
from .wrapperdefs import API_PREFIX_UNMARSHAL
from .wrapperdefs import VULKAN_STREAM_TYPE
from copy import copy
from dataclasses import dataclass
decoder_snapshot_decl_preamble = """
namespace android {
namespace base {
class BumpPool;
class Stream;
} // namespace base {
} // namespace android {
class VkDecoderSnapshot {
public:
VkDecoderSnapshot();
~VkDecoderSnapshot();
void save(android::base::Stream* stream);
void load(android::base::Stream* stream, emugl::GfxApiLogger& gfx_logger,
emugl::HealthMonitor<>* healthMonitor);
void createExtraHandlesForNextApi(const uint64_t* created, uint32_t count);
"""
decoder_snapshot_decl_postamble = """
private:
class Impl;
std::unique_ptr<Impl> mImpl;
};
"""
decoder_snapshot_impl_preamble ="""
using namespace gfxstream::vk;
using emugl::GfxApiLogger;
using emugl::HealthMonitor;
class VkDecoderSnapshot::Impl {
public:
Impl() { }
void save(android::base::Stream* stream) {
mReconstruction.save(stream);
}
void load(android::base::Stream* stream, GfxApiLogger& gfx_logger,
HealthMonitor<>* healthMonitor) {
mReconstruction.load(stream, gfx_logger, healthMonitor);
}
void createExtraHandlesForNextApi(const uint64_t* created, uint32_t count) {
mLock.lock();
mReconstruction.createExtraHandlesForNextApi(created, count);
}
"""
decoder_snapshot_impl_postamble = """
private:
android::base::Lock mLock;
VkReconstruction mReconstruction;
};
VkDecoderSnapshot::VkDecoderSnapshot() :
mImpl(new VkDecoderSnapshot::Impl()) { }
void VkDecoderSnapshot::save(android::base::Stream* stream) {
mImpl->save(stream);
}
void VkDecoderSnapshot::load(android::base::Stream* stream, GfxApiLogger& gfx_logger,
HealthMonitor<>* healthMonitor) {
mImpl->load(stream, gfx_logger, healthMonitor);
}
void VkDecoderSnapshot::createExtraHandlesForNextApi(const uint64_t* created, uint32_t count) {
mImpl->createExtraHandlesForNextApi(created, count);
}
VkDecoderSnapshot::~VkDecoderSnapshot() = default;
"""
AUXILIARY_SNAPSHOT_API_BASE_PARAM_COUNT = 3
AUXILIARY_SNAPSHOT_API_PARAM_NAMES = [
"input_result",
]
# Vulkan handle dependencies.
# (a, b): a depends on b
SNAPSHOT_HANDLE_DEPENDENCIES = [
# Dispatchable handle types
("VkCommandBuffer", "VkCommandPool"),
("VkCommandPool", "VkDevice"),
("VkQueue", "VkDevice"),
("VkDevice", "VkPhysicalDevice"),
("VkPhysicalDevice", "VkInstance")] + \
list(map(lambda handleType : (handleType, "VkDevice"), NON_DISPATCHABLE_HANDLE_TYPES))
handleDependenciesDict = dict(SNAPSHOT_HANDLE_DEPENDENCIES)
def extract_deps_vkAllocateMemory(param, access, lenExpr, api, cgen):
cgen.stmt("const VkMemoryDedicatedAllocateInfo* dedicatedAllocateInfo = vk_find_struct<VkMemoryDedicatedAllocateInfo>(pAllocateInfo)");
cgen.beginIf("dedicatedAllocateInfo");
cgen.beginIf("dedicatedAllocateInfo->image")
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)%s, %s, (uint64_t)(uintptr_t)%s)" % \
(access, lenExpr, "unboxed_to_boxed_non_dispatchable_VkImage(dedicatedAllocateInfo->image)"))
cgen.endIf()
cgen.beginIf("dedicatedAllocateInfo->buffer")
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)%s, %s, (uint64_t)(uintptr_t)%s)" % \
(access, lenExpr, "unboxed_to_boxed_non_dispatchable_VkBuffer(dedicatedAllocateInfo->buffer)"))
cgen.endIf()
cgen.endIf()
def extract_deps_vkAllocateCommandBuffers(param, access, lenExpr, api, cgen):
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)%s, %s, (uint64_t)(uintptr_t)%s)" % \
(access, lenExpr, "unboxed_to_boxed_non_dispatchable_VkCommandPool(pAllocateInfo->commandPool)"))
def extract_deps_vkCreateImageView(param, access, lenExpr, api, cgen):
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)%s, %s, (uint64_t)(uintptr_t)%s, VkReconstruction::CREATED, VkReconstruction::BOUND_MEMORY)" % \
(access, lenExpr, "unboxed_to_boxed_non_dispatchable_VkImage(pCreateInfo->image)"))
def extract_deps_vkCreateGraphicsPipelines(param, access, lenExpr, api, cgen):
cgen.beginFor("uint32_t i = 0", "i < createInfoCount", "++i")
cgen.beginFor("uint32_t j = 0", "j < pCreateInfos[i].stageCount", "++j")
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)(%s + i), %s, (uint64_t)(uintptr_t)%s)" % \
(access, 1, "unboxed_to_boxed_non_dispatchable_VkShaderModule(pCreateInfos[i].pStages[j].module)"))
cgen.endFor()
cgen.endFor()
def extract_deps_vkCreateFramebuffer(param, access, lenExpr, api, cgen):
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)%s, %s, (uint64_t)(uintptr_t)%s)" % \
(access, lenExpr, "unboxed_to_boxed_non_dispatchable_VkRenderPass(pCreateInfo->renderPass)"))
cgen.beginFor("uint32_t i = 0", "i < pCreateInfo->attachmentCount" , "++i")
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)%s, %s, (uint64_t)(uintptr_t)%s)" % \
(access, lenExpr, "unboxed_to_boxed_non_dispatchable_VkImageView(pCreateInfo->pAttachments[i])"))
cgen.endFor()
def extract_deps_vkBindImageMemory(param, access, lenExpr, api, cgen):
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)%s, %s, (uint64_t)(uintptr_t)%s, VkReconstruction::BOUND_MEMORY)" % \
(access, lenExpr, "unboxed_to_boxed_non_dispatchable_VkDeviceMemory(memory)"))
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)%s, %s, (uint64_t)(uintptr_t)((%s)[0]), VkReconstruction::BOUND_MEMORY)" % \
(access, lenExpr, access))
def extract_deps_vkBindBufferMemory(param, access, lenExpr, api, cgen):
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)%s, %s, (uint64_t)(uintptr_t)%s, VkReconstruction::BOUND_MEMORY)" % \
(access, lenExpr, "unboxed_to_boxed_non_dispatchable_VkDeviceMemory(memory)"))
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)%s, %s, (uint64_t)(uintptr_t)((%s)[0]), VkReconstruction::BOUND_MEMORY)" % \
(access, lenExpr, access))
specialCaseDependencyExtractors = {
"vkAllocateCommandBuffers" : extract_deps_vkAllocateCommandBuffers,
"vkAllocateMemory" : extract_deps_vkAllocateMemory,
"vkCreateImageView" : extract_deps_vkCreateImageView,
"vkCreateGraphicsPipelines" : extract_deps_vkCreateGraphicsPipelines,
"vkCreateFramebuffer" : extract_deps_vkCreateFramebuffer,
"vkBindImageMemory": extract_deps_vkBindImageMemory,
"vkBindBufferMemory": extract_deps_vkBindBufferMemory,
}
apiSequences = {
"vkAllocateMemory" : ["vkAllocateMemory", "vkMapMemoryIntoAddressSpaceGOOGLE"]
}
apiCrreateExtraHandles = [
"vkCreateDevice",
"vkCreateDescriptorPool",
]
@dataclass(frozen=True)
class VkObjectState:
vk_object : str
state : str = "VkReconstruction::CREATED"
# TODO: add vkBindImageMemory2 and vkBindBufferMemory2 into this list
apiChangeState = {
"vkBindImageMemory": VkObjectState("image", "VkReconstruction::BOUND_MEMORY"),
"vkBindBufferMemory": VkObjectState("buffer", "VkReconstruction::BOUND_MEMORY"),
}
def api_special_implementation_vkBindImageMemory2(api, cgen):
childType = "VkImage"
parentType = "VkDeviceMemory"
childObj = "boxed_%s" % childType
parentObj = "boxed_%s" % parentType
cgen.stmt("android::base::AutoLock lock(mLock)")
cgen.beginFor("uint32_t i = 0", "i < bindInfoCount", "++i")
cgen.stmt("%s boxed_%s = unboxed_to_boxed_non_dispatchable_%s(pBindInfos[i].image)"
% (childType, childType, childType))
cgen.stmt("%s boxed_%s = unboxed_to_boxed_non_dispatchable_%s(pBindInfos[i].memory)"
% (parentType, parentType, parentType))
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)&%s, %s, (uint64_t)(uintptr_t)%s, VkReconstruction::BOUND_MEMORY)" % \
(childObj, "1", parentObj))
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)&%s, %s, (uint64_t)(uintptr_t)%s, VkReconstruction::BOUND_MEMORY)" % \
(childObj, "1", childObj))
cgen.endFor()
cgen.stmt("auto apiHandle = mReconstruction.createApiInfo()")
cgen.stmt("auto apiInfo = mReconstruction.getApiInfo(apiHandle)")
cgen.stmt("mReconstruction.setApiTrace(apiInfo, OP_%s, snapshotTraceBegin, snapshotTraceBytes)" % api.name)
cgen.line("// Note: the implementation does not work with bindInfoCount > 1");
cgen.beginFor("uint32_t i = 0", "i < bindInfoCount", "++i")
cgen.stmt("%s boxed_%s = unboxed_to_boxed_non_dispatchable_%s(pBindInfos[i].image)"
% (childType, childType, childType))
cgen.stmt(f"mReconstruction.forEachHandleAddApi((const uint64_t*)&{childObj}, {1}, apiHandle, VkReconstruction::BOUND_MEMORY)")
cgen.endFor()
apiSpecialImplementation = {
"vkBindImageMemory2": api_special_implementation_vkBindImageMemory2,
"vkBindImageMemory2KHR": api_special_implementation_vkBindImageMemory2,
}
apiModifies = {
"vkMapMemoryIntoAddressSpaceGOOGLE" : ["memory"],
"vkGetBlobGOOGLE" : ["memory"],
"vkBeginCommandBuffer" : ["commandBuffer"],
"vkEndCommandBuffer" : ["commandBuffer"],
}
apiClearModifiers = {
"vkResetCommandBuffer" : ["commandBuffer"],
}
delayedDestroys = [
"vkDestroyShaderModule",
]
# The following types are created and cached by other commands.
# Thus we should not snapshot their "create" commands.
skipCreatorSnapshotTypes = [
"VkQueue", # created by vkCreateDevice
"VkDescriptorSet", # created by vkCreateDescriptorPool
]
def is_state_change_operation(api, param):
if param.isCreatedBy(api) and param.typeName not in skipCreatorSnapshotTypes:
return True
if api.name in apiChangeState:
if param.paramName == apiChangeState[api.name].vk_object:
return True
return False
def get_target_state(api, param):
if param.isCreatedBy(api):
return "VkReconstruction::CREATED"
if api.name in apiChangeState:
if param.paramName == apiChangeState[api.name].vk_object:
return apiChangeState[api.name].state
return None
def is_modify_operation(api, param):
if api.name in apiModifies:
if param.paramName in apiModifies[api.name]:
return True
if api.name.startswith('vkCmd') and param.paramName == 'commandBuffer':
return True
return False
def is_clear_modifier_operation(api, param):
if api.name in apiClearModifiers:
if param.paramName in apiClearModifiers[api.name]:
return True
def emit_impl(typeInfo, api, cgen):
if api.name in apiSpecialImplementation:
apiSpecialImplementation[api.name](api, cgen)
for p in api.parameters:
if not (p.isHandleType):
continue
lenExpr = cgen.generalLengthAccess(p)
lenAccessGuard = cgen.generalLengthAccessGuard(p)
if lenExpr is None:
lenExpr = "1"
# Note that in vkCreate*, the last parameter (the output) is boxed. But all input parameters are unboxed.
if p.pointerIndirectionLevels > 0:
access = p.paramName
else:
access = "(&%s)" % p.paramName
if is_state_change_operation(api, p):
if p.isCreatedBy(api):
boxed_access = access
else:
cgen.stmt("%s boxed_%s = unboxed_to_boxed_non_dispatchable_%s(%s[0])" % (p.typeName, p.typeName, p.typeName, access))
boxed_access = "&boxed_%s" % p.typeName
if p.pointerIndirectionLevels > 0:
cgen.stmt("if (!%s) return" % access)
isCreateExtraHandleApi = api.name in apiCrreateExtraHandles
if not isCreateExtraHandleApi:
cgen.stmt("android::base::AutoLock lock(mLock)")
cgen.line("// %s create" % p.paramName)
if p.isCreatedBy(api):
cgen.stmt("mReconstruction.addHandles((const uint64_t*)%s, %s)" % (boxed_access, lenExpr));
if p.isCreatedBy(api) and p.typeName in handleDependenciesDict:
dependsOnType = handleDependenciesDict[p.typeName];
for p2 in api.parameters:
if p2.typeName == dependsOnType:
cgen.stmt("mReconstruction.addHandleDependency((const uint64_t*)%s, %s, (uint64_t)(uintptr_t)%s)" % (boxed_access, lenExpr, p2.paramName))
if api.name in specialCaseDependencyExtractors:
specialCaseDependencyExtractors[api.name](p, boxed_access, lenExpr, api, cgen)
cgen.stmt("auto apiHandle = mReconstruction.createApiInfo()")
cgen.stmt("auto apiInfo = mReconstruction.getApiInfo(apiHandle)")
cgen.stmt("mReconstruction.setApiTrace(apiInfo, OP_%s, snapshotTraceBegin, snapshotTraceBytes)" % api.name)
if lenAccessGuard is not None:
cgen.beginIf(lenAccessGuard)
cgen.stmt(f"mReconstruction.forEachHandleAddApi((const uint64_t*){boxed_access}, {lenExpr}, apiHandle, {get_target_state(api, p)})")
if p.isCreatedBy(api):
cgen.stmt("mReconstruction.setCreatedHandlesForApi(apiHandle, (const uint64_t*)%s, %s)" % (boxed_access, lenExpr))
if lenAccessGuard is not None:
cgen.endIf()
if isCreateExtraHandleApi:
cgen.stmt("mLock.unlock()")
if p.isDestroyedBy(api):
cgen.stmt("android::base::AutoLock lock(mLock)")
cgen.line("// %s destroy" % p.paramName)
if lenAccessGuard is not None:
cgen.beginIf(lenAccessGuard)
shouldRecursiveDestroy = "false" if api.name in delayedDestroys else "true"
cgen.stmt("mReconstruction.removeHandles((const uint64_t*)%s, %s, %s)" % (access, lenExpr, shouldRecursiveDestroy));
if lenAccessGuard is not None:
cgen.endIf()
if is_modify_operation(api, p) or is_clear_modifier_operation(api, p):
cgen.stmt("android::base::AutoLock lock(mLock)")
cgen.line("// %s modify" % p.paramName)
cgen.stmt("auto apiHandle = mReconstruction.createApiInfo()")
cgen.stmt("auto apiInfo = mReconstruction.getApiInfo(apiHandle)")
cgen.stmt("mReconstruction.setApiTrace(apiInfo, OP_%s, snapshotTraceBegin, snapshotTraceBytes)" % api.name)
if lenAccessGuard is not None:
cgen.beginIf(lenAccessGuard)
cgen.beginFor("uint32_t i = 0", "i < %s" % lenExpr, "++i")
if p.isNonDispatchableHandleType():
cgen.stmt("%s boxed = unboxed_to_boxed_non_dispatchable_%s(%s[i])" % (p.typeName, p.typeName, access))
else:
cgen.stmt("%s boxed = unboxed_to_boxed_%s(%s[i])" % (p.typeName, p.typeName, access))
if is_modify_operation(api, p):
cgen.stmt("mReconstruction.forEachHandleAddModifyApi((const uint64_t*)(&boxed), 1, apiHandle)")
else: # is clear modifier operation
cgen.stmt("mReconstruction.forEachHandleClearModifyApi((const uint64_t*)(&boxed), 1)")
cgen.endFor()
if lenAccessGuard is not None:
cgen.endIf()
def emit_passthrough_to_impl(typeInfo, api, cgen):
cgen.vkApiCall(api, customPrefix = "mImpl->")
class VulkanDecoderSnapshot(VulkanWrapperGenerator):
def __init__(self, module, typeInfo):
VulkanWrapperGenerator.__init__(self, module, typeInfo)
self.typeInfo = typeInfo
self.cgenHeader = CodeGen()
self.cgenHeader.incrIndent()
self.cgenImpl = CodeGen()
self.currentFeature = None
self.feature_apis = []
def onBegin(self,):
self.module.appendHeader(decoder_snapshot_decl_preamble)
self.module.appendImpl(decoder_snapshot_impl_preamble)
def onBeginFeature(self, featureName, featureType):
VulkanWrapperGenerator.onBeginFeature(self, featureName, featureType)
self.currentFeature = featureName
def onGenCmd(self, cmdinfo, name, alias):
VulkanWrapperGenerator.onGenCmd(self, cmdinfo, name, alias)
api = self.typeInfo.apis[name]
additionalParams = [ \
makeVulkanTypeSimple(True, "uint8_t", 1, "snapshotTraceBegin"),
makeVulkanTypeSimple(False, "size_t", 0, "snapshotTraceBytes"),
makeVulkanTypeSimple(False, "android::base::BumpPool", 1, "pool"),]
if api.retType.typeName != "void":
additionalParams.append( \
makeVulkanTypeSimple(False, api.retType.typeName, 0, "input_result"))
apiForSnapshot = \
api.withCustomParameters( \
additionalParams + \
api.parameters).withCustomReturnType( \
makeVulkanTypeSimple(False, "void", 0, "void"))
self.feature_apis.append((self.currentFeature, apiForSnapshot))
self.cgenHeader.stmt(self.cgenHeader.makeFuncProto(apiForSnapshot))
self.module.appendHeader(self.cgenHeader.swapCode())
self.cgenImpl.emitFuncImpl( \
apiForSnapshot, lambda cgen: emit_impl(self.typeInfo, apiForSnapshot, cgen))
self.module.appendImpl(self.cgenImpl.swapCode())
def onEnd(self,):
self.module.appendHeader(decoder_snapshot_decl_postamble)
self.module.appendImpl(decoder_snapshot_impl_postamble)
self.cgenHeader.decrIndent()
for feature, api in self.feature_apis:
if feature is not None:
self.cgenImpl.line("#ifdef %s" % feature)
apiImplShell = \
api.withModifiedName("VkDecoderSnapshot::" + api.name)
self.cgenImpl.emitFuncImpl( \
apiImplShell, lambda cgen: emit_passthrough_to_impl(self.typeInfo, api, cgen))
if feature is not None:
self.cgenImpl.line("#endif")
self.module.appendImpl(self.cgenImpl.swapCode())