/* * Copyright (C) 2018-2021 Intel Corporation * * SPDX-License-Identifier: MIT * */ #include "shared/source/os_interface/linux/os_context_linux.h" #include "shared/source/debug_settings/debug_settings_manager.h" #include "shared/source/execution_environment/execution_environment.h" #include "shared/source/execution_environment/root_device_environment.h" #include "shared/source/helpers/engine_node_helper.h" #include "shared/source/helpers/hw_helper.h" #include "shared/source/helpers/hw_info.h" #include "shared/source/os_interface/linux/drm_neo.h" #include "shared/source/os_interface/os_context.h" #include "shared/source/os_interface/os_interface.h" namespace NEO { OsContext *OsContext::create(OSInterface *osInterface, uint32_t contextId, DeviceBitfield deviceBitfield, EngineTypeUsage typeUsage, PreemptionMode preemptionMode, bool rootDevice) { if (osInterface) { return new OsContextLinux(*osInterface->getDriverModel()->as(), contextId, deviceBitfield, typeUsage, preemptionMode, rootDevice); } return new OsContext(contextId, deviceBitfield, typeUsage, preemptionMode, rootDevice); } OsContextLinux::OsContextLinux(Drm &drm, uint32_t contextId, DeviceBitfield deviceBitfield, EngineTypeUsage typeUsage, PreemptionMode preemptionMode, bool rootDevice) : OsContext(contextId, deviceBitfield, typeUsage, preemptionMode, rootDevice), drm(drm) {} void OsContextLinux::initializeContext() { auto hwInfo = drm.getRootDeviceEnvironment().getHardwareInfo(); auto defaultEngineType = getChosenEngineType(*hwInfo); if (engineType == defaultEngineType && !isLowPriority() && !isInternalEngine()) { this->setDefaultContext(true); } bool submitDirect = false; this->isDirectSubmissionAvailable(*drm.getRootDeviceEnvironment().getHardwareInfo(), submitDirect); if (drm.isPerContextVMRequired()) { this->drmVmIds.resize(deviceBitfield.size(), 0); } for (auto deviceIndex = 0u; deviceIndex < deviceBitfield.size(); deviceIndex++) { if (deviceBitfield.test(deviceIndex)) { auto drmVmId = drm.getVirtualMemoryAddressSpace(deviceIndex); auto drmContextId = drm.createDrmContext(drmVmId, drm.isVmBindAvailable()); if (drm.areNonPersistentContextsSupported()) { drm.setNonPersistentContext(drmContextId); } if (drm.getRootDeviceEnvironment().executionEnvironment.isDebuggingEnabled() && !isInternalEngine()) { drm.setContextDebugFlag(drmContextId); } if (drm.isPreemptionSupported() && isLowPriority()) { drm.setLowPriorityContextParam(drmContextId); } this->engineFlag = drm.bindDrmContext(drmContextId, deviceIndex, engineType, engineInstancedDevice); this->drmContextIds.push_back(drmContextId); if (drm.isPerContextVMRequired()) { auto ret = drm.queryVmId(drmContextId, drmVmId); DEBUG_BREAK_IF(drmVmId == 0); DEBUG_BREAK_IF(ret != 0); UNUSED_VARIABLE(ret); UNRECOVERABLE_IF(this->drmVmIds.size() <= deviceIndex); this->drmVmIds[deviceIndex] = drmVmId; } } } } bool OsContextLinux::isDirectSubmissionSupported(const HardwareInfo &hwInfo) const { auto &hwHelper = HwHelper::get(hwInfo.platform.eRenderCoreFamily); return this->getDrm().isVmBindAvailable() && hwHelper.isDirectSubmissionSupported(); } Drm &OsContextLinux::getDrm() const { return this->drm; } void OsContextLinux::waitForPagingFence() { for (auto drmIterator = 0u; drmIterator < this->deviceBitfield.size(); drmIterator++) { if (this->deviceBitfield.test(drmIterator)) { drm.waitForBind(drmIterator); } } } OsContextLinux::~OsContextLinux() { if (contextInitialized) { for (auto drmContextId : drmContextIds) { drm.destroyDrmContext(drmContextId); } } } } // namespace NEO