//============================================================================ // Copyright (c) Kitware, Inc. // All rights reserved. // See LICENSE.txt for details. // // This software is distributed WITHOUT ANY WARRANTY; without even // the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR // PURPOSE. See the above copyright notice for more information. //============================================================================ #include #include #include #include #include #include #include //Bring in each device adapters runtime class #include #include #include #include #include #include //for tolower namespace { class DeviceAdapterMemoryManagerInvalid : public vtkm::cont::internal::DeviceAdapterMemoryManagerBase { public: VTKM_CONT virtual ~DeviceAdapterMemoryManagerInvalid() override {} VTKM_CONT virtual vtkm::cont::internal::BufferInfo Allocate(vtkm::BufferSizeType) const override { throw vtkm::cont::ErrorBadDevice("Tried to manage memory on an invalid device."); } VTKM_CONT virtual vtkm::cont::DeviceAdapterId GetDevice() const override { return vtkm::cont::DeviceAdapterTagUndefined{}; } VTKM_CONT virtual vtkm::cont::internal::BufferInfo CopyHostToDevice( const vtkm::cont::internal::BufferInfo&) const override { throw vtkm::cont::ErrorBadDevice("Tried to manage memory on an invalid device."); } VTKM_CONT virtual void CopyHostToDevice(const vtkm::cont::internal::BufferInfo&, const vtkm::cont::internal::BufferInfo&) const override { throw vtkm::cont::ErrorBadDevice("Tried to manage memory on an invalid device."); } VTKM_CONT virtual vtkm::cont::internal::BufferInfo CopyDeviceToHost( const vtkm::cont::internal::BufferInfo&) const override { throw vtkm::cont::ErrorBadDevice("Tried to manage memory on an invalid device."); } VTKM_CONT virtual void CopyDeviceToHost(const vtkm::cont::internal::BufferInfo&, const vtkm::cont::internal::BufferInfo&) const override { throw vtkm::cont::ErrorBadDevice("Tried to manage memory on an invalid device."); } VTKM_CONT virtual vtkm::cont::internal::BufferInfo CopyDeviceToDevice( const vtkm::cont::internal::BufferInfo&) const override { throw vtkm::cont::ErrorBadDevice("Tried to manage memory on an invalid device."); } VTKM_CONT virtual void CopyDeviceToDevice(const vtkm::cont::internal::BufferInfo&, const vtkm::cont::internal::BufferInfo&) const override { throw vtkm::cont::ErrorBadDevice("Tried to manage memory on an invalid device."); } }; class RuntimeDeviceConfigurationInvalid final : public vtkm::cont::internal::RuntimeDeviceConfigurationBase { public: VTKM_CONT virtual ~RuntimeDeviceConfigurationInvalid() override final {} VTKM_CONT virtual vtkm::cont::DeviceAdapterId GetDevice() const override final { return vtkm::cont::DeviceAdapterTagUndefined{}; } VTKM_CONT virtual vtkm::cont::internal::RuntimeDeviceConfigReturnCode SetThreads( const vtkm::Id&) override final { throw vtkm::cont::ErrorBadDevice("Tried to set the number of threads on an invalid device"); } VTKM_CONT virtual vtkm::cont::internal::RuntimeDeviceConfigReturnCode SetNumaRegions( const vtkm::Id&) override final { throw vtkm::cont::ErrorBadDevice( "Tried to set the number of numa regions on an invalid device"); } VTKM_CONT virtual vtkm::cont::internal::RuntimeDeviceConfigReturnCode SetDeviceInstance( const vtkm::Id&) override final { throw vtkm::cont::ErrorBadDevice("Tried to set the device instance on an invalid device"); } VTKM_CONT virtual vtkm::cont::internal::RuntimeDeviceConfigReturnCode GetThreads( vtkm::Id&) const override final { throw vtkm::cont::ErrorBadDevice("Tried to get the number of threads on an invalid device"); } VTKM_CONT virtual vtkm::cont::internal::RuntimeDeviceConfigReturnCode GetNumaRegions( vtkm::Id&) const override final { throw vtkm::cont::ErrorBadDevice( "Tried to get the number of numa regions on an invalid device"); } VTKM_CONT virtual vtkm::cont::internal::RuntimeDeviceConfigReturnCode GetDeviceInstance( vtkm::Id&) const override final { throw vtkm::cont::ErrorBadDevice("Tried to get the device instance on an invalid device"); } VTKM_CONT virtual vtkm::cont::internal::RuntimeDeviceConfigReturnCode GetMaxThreads( vtkm::Id&) const override final { throw vtkm::cont::ErrorBadDevice("Tried to get the max number of threads on an invalid device"); } VTKM_CONT virtual vtkm::cont::internal::RuntimeDeviceConfigReturnCode GetMaxDevices( vtkm::Id&) const override final { throw vtkm::cont::ErrorBadDevice("Tried to get the max number of devices on an invalid device"); } }; struct VTKM_NEVER_EXPORT InitializeDeviceNames { vtkm::cont::DeviceAdapterNameType* Names; vtkm::cont::DeviceAdapterNameType* LowerCaseNames; VTKM_CONT InitializeDeviceNames(vtkm::cont::DeviceAdapterNameType* names, vtkm::cont::DeviceAdapterNameType* lower) : Names(names) , LowerCaseNames(lower) { std::fill_n(this->Names, VTKM_MAX_DEVICE_ADAPTER_ID, "InvalidDeviceId"); std::fill_n(this->LowerCaseNames, VTKM_MAX_DEVICE_ADAPTER_ID, "invaliddeviceid"); } template VTKM_CONT void operator()(Device device) { auto lowerCaseFunc = [](char c) { return static_cast(std::tolower(static_cast(c))); }; auto id = device.GetValue(); if (id > 0 && id < VTKM_MAX_DEVICE_ADAPTER_ID) { auto name = vtkm::cont::DeviceAdapterTraits::GetName(); this->Names[id] = name; std::transform(name.begin(), name.end(), name.begin(), lowerCaseFunc); this->LowerCaseNames[id] = name; } } }; struct VTKM_NEVER_EXPORT InitializeDeviceMemoryManagers { std::unique_ptr* Managers; VTKM_CONT InitializeDeviceMemoryManagers( std::unique_ptr* managers) : Managers(managers) { } template VTKM_CONT void CreateManager(Device device, std::true_type) { auto id = device.GetValue(); if (id > 0 && id < VTKM_MAX_DEVICE_ADAPTER_ID) { this->Managers[id].reset(new vtkm::cont::internal::DeviceAdapterMemoryManager); } } template VTKM_CONT void CreateManager(Device, std::false_type) { // No manager for invalid devices. } template VTKM_CONT void operator()(Device device) { this->CreateManager(device, std::integral_constant{}); } }; struct VTKM_NEVER_EXPORT InitializeRuntimeDeviceConfigurations { std::unique_ptr* RuntimeConfigurations; vtkm::cont::internal::RuntimeDeviceConfigurationOptions RuntimeConfigurationOptions; VTKM_CONT InitializeRuntimeDeviceConfigurations( std::unique_ptr* runtimeConfigurations, const vtkm::cont::internal::RuntimeDeviceConfigurationOptions& configOptions) : RuntimeConfigurations(runtimeConfigurations) , RuntimeConfigurationOptions(configOptions) { if (!configOptions.IsInitialized()) { VTKM_LOG_S(vtkm::cont::LogLevel::Warn, "Initializing 'RuntimeDeviceConfigurations' with uninitialized configOptions. Did " "you call vtkm::cont::Initialize?"); } } template VTKM_CONT void CreateRuntimeConfiguration(Device device, int& argc, char* argv[], std::true_type) { auto id = device.GetValue(); if (id > 0 && id < VTKM_MAX_DEVICE_ADAPTER_ID) { this->RuntimeConfigurations[id].reset( new vtkm::cont::internal::RuntimeDeviceConfiguration); this->RuntimeConfigurations[id]->Initialize(RuntimeConfigurationOptions, argc, argv); } } template VTKM_CONT void CreateRuntimeConfiguration(Device, int&, char**, std::false_type) { // No runtime configuration for invalid devices. } template VTKM_CONT void operator()(Device device, int& argc, char* argv[]) { this->CreateRuntimeConfiguration( device, argc, argv, std::integral_constant{}); } }; struct VTKM_NEVER_EXPORT RuntimeDeviceInformationFunctor { bool Exists = false; template VTKM_CONT void operator()(DeviceAdapter, vtkm::cont::DeviceAdapterId device) { if (DeviceAdapter() == device) { this->Exists = vtkm::cont::DeviceAdapterRuntimeDetector().Exists(); } } }; class RuntimeDeviceNames { public: static const vtkm::cont::DeviceAdapterNameType& GetDeviceName(vtkm::Int8 id) { return Instance().DeviceNames[id]; } static const vtkm::cont::DeviceAdapterNameType& GetLowerCaseDeviceName(vtkm::Int8 id) { return Instance().LowerCaseDeviceNames[id]; } private: static const RuntimeDeviceNames& Instance() { static RuntimeDeviceNames instance; return instance; } RuntimeDeviceNames() { InitializeDeviceNames functor(DeviceNames, LowerCaseDeviceNames); vtkm::ListForEach(functor, VTKM_DEFAULT_DEVICE_ADAPTER_LIST()); } friend struct InitializeDeviceNames; vtkm::cont::DeviceAdapterNameType DeviceNames[VTKM_MAX_DEVICE_ADAPTER_ID]; vtkm::cont::DeviceAdapterNameType LowerCaseDeviceNames[VTKM_MAX_DEVICE_ADAPTER_ID]; }; class RuntimeDeviceMemoryManagers { public: static vtkm::cont::internal::DeviceAdapterMemoryManagerBase& GetDeviceMemoryManager( vtkm::cont::DeviceAdapterId device) { const auto id = device.GetValue(); if (device.IsValueValid()) { auto&& manager = Instance().DeviceMemoryManagers[id]; if (manager) { return *manager.get(); } else { return Instance().InvalidManager; } } else { return Instance().InvalidManager; } } private: static RuntimeDeviceMemoryManagers& Instance() { static RuntimeDeviceMemoryManagers instance; return instance; } RuntimeDeviceMemoryManagers() { InitializeDeviceMemoryManagers functor(this->DeviceMemoryManagers); vtkm::ListForEach(functor, VTKM_DEFAULT_DEVICE_ADAPTER_LIST()); } friend struct InitializeDeviceMemoryManagers; std::unique_ptr DeviceMemoryManagers[VTKM_MAX_DEVICE_ADAPTER_ID]; DeviceAdapterMemoryManagerInvalid InvalidManager; }; class RuntimeDeviceConfigurations { public: static vtkm::cont::internal::RuntimeDeviceConfigurationBase& GetRuntimeDeviceConfiguration( vtkm::cont::DeviceAdapterId device, const vtkm::cont::internal::RuntimeDeviceConfigurationOptions& configOptions, int& argc, char* argv[]) { const auto id = device.GetValue(); if (device.IsValueValid()) { auto&& runtimeConfiguration = Instance(configOptions, argc, argv).DeviceConfigurations[id]; if (runtimeConfiguration) { return *runtimeConfiguration.get(); } else { return Instance(configOptions, argc, argv).InvalidConfiguration; } } else { return Instance(configOptions, argc, argv).InvalidConfiguration; } } private: static RuntimeDeviceConfigurations& Instance( const vtkm::cont::internal::RuntimeDeviceConfigurationOptions& configOptions, int& argc, char* argv[]) { static RuntimeDeviceConfigurations instance{ configOptions, argc, argv }; return instance; } RuntimeDeviceConfigurations( const vtkm::cont::internal::RuntimeDeviceConfigurationOptions& configOptions, int& argc, char* argv[]) { InitializeRuntimeDeviceConfigurations functor(this->DeviceConfigurations, configOptions); vtkm::ListForEach(functor, VTKM_DEFAULT_DEVICE_ADAPTER_LIST(), argc, argv); } friend struct InitializeRuntimeDeviceConfigurations; std::unique_ptr DeviceConfigurations[VTKM_MAX_DEVICE_ADAPTER_ID]; RuntimeDeviceConfigurationInvalid InvalidConfiguration; }; } // namespace namespace vtkm { namespace cont { namespace detail { } VTKM_CONT DeviceAdapterNameType RuntimeDeviceInformation::GetName(DeviceAdapterId device) const { const auto id = device.GetValue(); if (device.IsValueValid()) { return RuntimeDeviceNames::GetDeviceName(id); } else if (id == VTKM_DEVICE_ADAPTER_UNDEFINED) { return vtkm::cont::DeviceAdapterTraits::GetName(); } else if (id == VTKM_DEVICE_ADAPTER_ANY) { return vtkm::cont::DeviceAdapterTraits::GetName(); } // Deviceis invalid: return RuntimeDeviceNames::GetDeviceName(0); } VTKM_CONT DeviceAdapterId RuntimeDeviceInformation::GetId(DeviceAdapterNameType name) const { // The GetDeviceAdapterId call is case-insensitive so transform the name to be lower case // as that is how we cache the case-insensitive version. auto lowerCaseFunc = [](char c) { return static_cast(std::tolower(static_cast(c))); }; std::transform(name.begin(), name.end(), name.begin(), lowerCaseFunc); //lower-case the name here if (name == "any") { return vtkm::cont::DeviceAdapterTagAny{}; } else if (name == "undefined") { return vtkm::cont::DeviceAdapterTagUndefined{}; } for (vtkm::Int8 id = 0; id < VTKM_MAX_DEVICE_ADAPTER_ID; ++id) { if (name == RuntimeDeviceNames::GetLowerCaseDeviceName(id)) { return vtkm::cont::make_DeviceAdapterId(id); } } return vtkm::cont::DeviceAdapterTagUndefined{}; } VTKM_CONT bool RuntimeDeviceInformation::Exists(DeviceAdapterId id) const { if (id == vtkm::cont::DeviceAdapterTagAny{}) { return true; } RuntimeDeviceInformationFunctor functor; vtkm::ListForEach(functor, VTKM_DEFAULT_DEVICE_ADAPTER_LIST(), id); return functor.Exists; } VTKM_CONT vtkm::cont::internal::DeviceAdapterMemoryManagerBase& RuntimeDeviceInformation::GetMemoryManager(DeviceAdapterId device) const { if (device.IsValueValid()) { return RuntimeDeviceMemoryManagers::GetDeviceMemoryManager(device); } else { throw vtkm::cont::ErrorBadValue( "Attempted to get a DeviceAdapterMemoryManager for an invalid device '" + device.GetName() + "'"); } } VTKM_CONT vtkm::cont::internal::RuntimeDeviceConfigurationBase& RuntimeDeviceInformation::GetRuntimeConfiguration( DeviceAdapterId device, const vtkm::cont::internal::RuntimeDeviceConfigurationOptions& configOptions, int& argc, char* argv[]) const { return RuntimeDeviceConfigurations::GetRuntimeDeviceConfiguration( device, configOptions, argc, argv); } VTKM_CONT vtkm::cont::internal::RuntimeDeviceConfigurationBase& RuntimeDeviceInformation::GetRuntimeConfiguration( DeviceAdapterId device, const vtkm::cont::internal::RuntimeDeviceConfigurationOptions& configOptions) const { int placeholder; return this->GetRuntimeConfiguration(device, configOptions, placeholder, nullptr); } VTKM_CONT vtkm::cont::internal::RuntimeDeviceConfigurationBase& RuntimeDeviceInformation::GetRuntimeConfiguration(DeviceAdapterId device) const { vtkm::cont::internal::RuntimeDeviceConfigurationOptions placeholder; return this->GetRuntimeConfiguration(device, placeholder); } } // namespace vtkm::cont } // namespace vtkm