2020-03-20 22:10:19 +00:00
|
|
|
/*
|
|
|
|
* Modern effects for a modern Streamer
|
|
|
|
* Copyright (C) 2020 Michael Fabian Dirks
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
|
|
* (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "nvidia-cuda-context.hpp"
|
2020-11-08 05:47:47 +00:00
|
|
|
#include <cassert>
|
2020-03-20 22:10:19 +00:00
|
|
|
#include <stdexcept>
|
2020-11-07 08:00:12 +00:00
|
|
|
#include "util/util-logging.hpp"
|
|
|
|
|
|
|
|
#ifdef _DEBUG
|
|
|
|
#define ST_PREFIX "<%s> "
|
|
|
|
#define D_LOG_ERROR(x, ...) P_LOG_ERROR(ST_PREFIX##x, __FUNCTION_SIG__, __VA_ARGS__)
|
|
|
|
#define D_LOG_WARNING(x, ...) P_LOG_WARN(ST_PREFIX##x, __FUNCTION_SIG__, __VA_ARGS__)
|
|
|
|
#define D_LOG_INFO(x, ...) P_LOG_INFO(ST_PREFIX##x, __FUNCTION_SIG__, __VA_ARGS__)
|
|
|
|
#define D_LOG_DEBUG(x, ...) P_LOG_DEBUG(ST_PREFIX##x, __FUNCTION_SIG__, __VA_ARGS__)
|
|
|
|
#else
|
|
|
|
#define ST_PREFIX "<nvidia::cuda::context> "
|
|
|
|
#define D_LOG_ERROR(...) P_LOG_ERROR(ST_PREFIX __VA_ARGS__)
|
|
|
|
#define D_LOG_WARNING(...) P_LOG_WARN(ST_PREFIX __VA_ARGS__)
|
|
|
|
#define D_LOG_INFO(...) P_LOG_INFO(ST_PREFIX __VA_ARGS__)
|
|
|
|
#define D_LOG_DEBUG(...) P_LOG_DEBUG(ST_PREFIX __VA_ARGS__)
|
|
|
|
#endif
|
2020-03-20 22:10:19 +00:00
|
|
|
|
|
|
|
#ifdef WIN32
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(push)
|
2020-07-12 16:41:50 +00:00
|
|
|
#pragma warning(disable : 4191 4242 4244 4365 4777 4986 5039 5204)
|
2020-03-20 22:10:19 +00:00
|
|
|
#endif
|
|
|
|
#include <atlutil.h>
|
|
|
|
#ifdef _MSC_VER
|
|
|
|
#pragma warning(pop)
|
|
|
|
#endif
|
|
|
|
#endif
|
|
|
|
|
2020-11-08 05:47:47 +00:00
|
|
|
#define ENABLE_STACK_CHECKS
|
|
|
|
|
2021-06-16 10:16:35 +00:00
|
|
|
streamfx::nvidia::cuda::context::~context()
|
2020-03-20 22:10:19 +00:00
|
|
|
{
|
2020-11-07 08:00:12 +00:00
|
|
|
D_LOG_DEBUG("Finalizing... (Addr: 0x%" PRIuPTR ")", this);
|
|
|
|
|
2020-03-20 22:10:19 +00:00
|
|
|
if (_has_device) {
|
|
|
|
_cuda->cuDevicePrimaryCtxRelease(_device);
|
|
|
|
}
|
|
|
|
_cuda->cuCtxDestroy(_ctx);
|
|
|
|
}
|
|
|
|
|
2021-06-16 10:16:35 +00:00
|
|
|
streamfx::nvidia::cuda::context::context()
|
|
|
|
: _cuda(::streamfx::nvidia::cuda::cuda::get()), _ctx(), _has_device(false), _device()
|
2020-11-08 05:47:47 +00:00
|
|
|
{
|
|
|
|
D_LOG_DEBUG("Initializating... (Addr: 0x%" PRIuPTR ")", this);
|
|
|
|
}
|
|
|
|
|
2020-03-20 22:10:19 +00:00
|
|
|
#ifdef WIN32
|
2021-06-16 10:16:35 +00:00
|
|
|
streamfx::nvidia::cuda::context::context(ID3D11Device* device) : context()
|
2020-03-20 22:10:19 +00:00
|
|
|
{
|
2021-06-16 10:16:35 +00:00
|
|
|
using namespace streamfx::nvidia::cuda;
|
2020-03-20 22:10:19 +00:00
|
|
|
|
|
|
|
if (!device)
|
|
|
|
throw std::invalid_argument("device");
|
|
|
|
// Get DXGI Device
|
|
|
|
IDXGIDevice* dxgi_device; // Don't use ATL::CComPtr
|
|
|
|
device->QueryInterface(__uuidof(IDXGIDevice), (void**)&dxgi_device);
|
|
|
|
|
|
|
|
// Get DXGI Adapter
|
|
|
|
ATL::CComPtr<IDXGIAdapter> dxgi_adapter;
|
|
|
|
dxgi_device->GetAdapter(&dxgi_adapter);
|
|
|
|
|
|
|
|
// Get Device Index
|
2020-06-14 17:17:26 +00:00
|
|
|
if (result res = _cuda->cuD3D11GetDevice(&_device, dxgi_adapter); res != result::SUCCESS) {
|
2020-03-20 22:10:19 +00:00
|
|
|
throw std::runtime_error("Failed to get device index for device.");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Acquire Context
|
2020-06-14 17:17:26 +00:00
|
|
|
if (result res = _cuda->cuDevicePrimaryCtxRetain(&_ctx, _device); res != result::SUCCESS) {
|
2020-03-20 22:10:19 +00:00
|
|
|
throw std::runtime_error("Failed to acquire primary device context.");
|
|
|
|
}
|
2020-04-11 01:12:39 +00:00
|
|
|
|
2020-03-20 22:10:19 +00:00
|
|
|
_has_device = true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2021-06-16 10:16:35 +00:00
|
|
|
::streamfx::nvidia::cuda::context_t streamfx::nvidia::cuda::context::get()
|
2020-03-20 22:10:19 +00:00
|
|
|
{
|
|
|
|
return _ctx;
|
|
|
|
}
|
2020-11-08 05:47:47 +00:00
|
|
|
|
2021-06-16 10:16:35 +00:00
|
|
|
std::shared_ptr<::streamfx::nvidia::cuda::context_stack> streamfx::nvidia::cuda::context::enter()
|
2020-11-08 05:47:47 +00:00
|
|
|
{
|
2021-06-16 10:16:35 +00:00
|
|
|
return std::make_shared<::streamfx::nvidia::cuda::context_stack>(shared_from_this());
|
2020-11-08 05:47:47 +00:00
|
|
|
}
|
|
|
|
|
2021-06-16 10:16:35 +00:00
|
|
|
void streamfx::nvidia::cuda::context::push()
|
2020-11-08 05:47:47 +00:00
|
|
|
{
|
2021-06-16 10:16:35 +00:00
|
|
|
if (auto res = _cuda->cuCtxPushCurrent(_ctx); res != ::streamfx::nvidia::cuda::result::SUCCESS) {
|
|
|
|
throw ::streamfx::nvidia::cuda::cuda_error(res);
|
2020-11-08 05:47:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-16 10:16:35 +00:00
|
|
|
void streamfx::nvidia::cuda::context::pop()
|
2020-11-08 05:47:47 +00:00
|
|
|
{
|
|
|
|
#ifdef ENABLE_STACK_CHECKS
|
2021-06-16 10:16:35 +00:00
|
|
|
::streamfx::nvidia::cuda::context_t ctx;
|
|
|
|
if (_cuda->cuCtxGetCurrent(&ctx) == ::streamfx::nvidia::cuda::result::SUCCESS)
|
2020-11-08 05:47:47 +00:00
|
|
|
assert(ctx == _ctx);
|
|
|
|
#endif
|
|
|
|
|
2021-06-16 10:16:35 +00:00
|
|
|
assert(_cuda->cuCtxPopCurrent(&ctx) == ::streamfx::nvidia::cuda::result::SUCCESS);
|
2020-11-08 05:47:47 +00:00
|
|
|
}
|
|
|
|
|
2021-06-16 10:16:35 +00:00
|
|
|
void streamfx::nvidia::cuda::context::synchronize()
|
2020-11-08 05:47:47 +00:00
|
|
|
{
|
|
|
|
D_LOG_DEBUG("Synchronizing... (Addr: 0x%" PRIuPTR ")", this);
|
|
|
|
|
|
|
|
#ifdef ENABLE_STACK_CHECKS
|
2021-06-16 10:16:35 +00:00
|
|
|
::streamfx::nvidia::cuda::context_t ctx;
|
|
|
|
if (_cuda->cuCtxGetCurrent(&ctx) == ::streamfx::nvidia::cuda::result::SUCCESS)
|
2020-11-08 05:47:47 +00:00
|
|
|
assert(ctx == _ctx);
|
|
|
|
#endif
|
|
|
|
|
2021-06-16 10:16:35 +00:00
|
|
|
if (auto res = _cuda->cuCtxSynchronize(); res != ::streamfx::nvidia::cuda::result::SUCCESS) {
|
|
|
|
throw ::streamfx::nvidia::cuda::cuda_error(res);
|
2020-11-08 05:47:47 +00:00
|
|
|
}
|
|
|
|
}
|