//============================================================================ // 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. // // Copyright 2016 Sandia Corporation. // Copyright 2016 UT-Battelle, LLC. // Copyright 2016 Los Alamos National Security. // // Under the terms of Contract DE-AC04-94AL85000 with Sandia Corporation, // the U.S. Government retains certain rights in this software. // // Under the terms of Contract DE-AC52-06NA25396 with Los Alamos National // Laboratory (LANL), the U.S. Government retains certain rights in // this software. //============================================================================ #include #include #include #include #include #include #include #include #include #define VTKM_MAX_DEVICE_ADAPTER_ID 8 namespace vtkm { namespace cont { namespace detail { struct RuntimeDeviceTrackerInternals { bool RuntimeValid[VTKM_MAX_DEVICE_ADAPTER_ID]; }; } VTKM_CONT RuntimeDeviceTracker::RuntimeDeviceTracker() : Internals(new detail::RuntimeDeviceTrackerInternals) { this->Reset(); } VTKM_CONT RuntimeDeviceTracker::~RuntimeDeviceTracker() { } VTKM_CONT void RuntimeDeviceTracker::CheckDevice(vtkm::cont::DeviceAdapterId deviceId, const vtkm::cont::DeviceAdapterNameType& deviceName) const { if ((deviceId < 0) || (deviceId >= VTKM_MAX_DEVICE_ADAPTER_ID)) { std::stringstream message; message << "Device '" << deviceName << "' has invalid ID of " << deviceId; throw vtkm::cont::ErrorBadValue(message.str()); } } VTKM_CONT bool RuntimeDeviceTracker::CanRunOnImpl(vtkm::cont::DeviceAdapterId deviceId, const vtkm::cont::DeviceAdapterNameType& deviceName) const { this->CheckDevice(deviceId, deviceName); return this->Internals->RuntimeValid[deviceId]; } VTKM_CONT void RuntimeDeviceTracker::SetDeviceState(vtkm::cont::DeviceAdapterId deviceId, const vtkm::cont::DeviceAdapterNameType& deviceName, bool state) { this->CheckDevice(deviceId, deviceName); this->Internals->RuntimeValid[deviceId] = state; } namespace { struct VTKM_NEVER_EXPORT RuntimeDeviceTrackerResetFunctor { vtkm::cont::RuntimeDeviceTracker Tracker; VTKM_CONT RuntimeDeviceTrackerResetFunctor(const vtkm::cont::RuntimeDeviceTracker& tracker) : Tracker(tracker) { } template VTKM_CONT void operator()(Device) { this->Tracker.ResetDevice(Device()); } }; } VTKM_CONT void RuntimeDeviceTracker::Reset() { std::fill_n(this->Internals->RuntimeValid, VTKM_MAX_DEVICE_ADAPTER_ID, false); RuntimeDeviceTrackerResetFunctor functor(*this); vtkm::ListForEach(functor, VTKM_DEFAULT_DEVICE_ADAPTER_LIST_TAG()); } VTKM_CONT vtkm::cont::RuntimeDeviceTracker RuntimeDeviceTracker::DeepCopy() const { vtkm::cont::RuntimeDeviceTracker dest; dest.DeepCopy(*this); return dest; } VTKM_CONT void RuntimeDeviceTracker::DeepCopy(const vtkm::cont::RuntimeDeviceTracker& src) { std::copy_n( src.Internals->RuntimeValid, VTKM_MAX_DEVICE_ADAPTER_ID, this->Internals->RuntimeValid); } VTKM_CONT void RuntimeDeviceTracker::ForceDeviceImpl(vtkm::cont::DeviceAdapterId deviceId, const vtkm::cont::DeviceAdapterNameType& deviceName, bool runtimeExists) { if (!runtimeExists) { std::stringstream message; message << "Cannot force to device '" << deviceName << "' because that device is not available on this system"; throw vtkm::cont::ErrorBadValue(message.str()); } this->CheckDevice(deviceId, deviceName); std::fill_n(this->Internals->RuntimeValid, VTKM_MAX_DEVICE_ADAPTER_ID, false); this->Internals->RuntimeValid[deviceId] = runtimeExists; } VTKM_CONT vtkm::cont::RuntimeDeviceTracker GetGlobalRuntimeDeviceTracker() { static vtkm::cont::RuntimeDeviceTracker globalTracker; return globalTracker; } } } // namespace vtkm::cont