/* XMRig * Copyright 2010 Jeff Garzik * Copyright 2012-2014 pooler * Copyright 2014 Lucas Jones * Copyright 2014-2016 Wolf9466 * Copyright 2016 Jay D Dee * Copyright 2017-2018 XMR-Stak , * Copyright 2018-2019 SChernykh * Copyright 2016-2019 XMRig , * * 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 3 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, see . */ #include "backend/cuda/CudaConfig.h" #include "backend/common/Tags.h" #include "backend/cuda/CudaConfig_gen.h" #include "backend/cuda/wrappers/CudaLib.h" #include "base/io/json/Json.h" #include "base/io/log/Log.h" #include "rapidjson/document.h" namespace xmrig { static bool generated = false; static const char *kDevicesHint = "devices-hint"; static const char *kEnabled = "enabled"; static const char *kLoader = "loader"; extern template class Threads; } rapidjson::Value xmrig::CudaConfig::toJSON(rapidjson::Document &doc) const { using namespace rapidjson; auto &allocator = doc.GetAllocator(); Value obj(kObjectType); obj.AddMember(StringRef(kEnabled), m_enabled, allocator); obj.AddMember(StringRef(kLoader), m_loader.toJSON(), allocator); m_threads.toJSON(obj, doc); return obj; } std::vector xmrig::CudaConfig::get(const Miner *miner, const Algorithm &algorithm, const std::vector &devices) const { std::vector out; const auto &threads = m_threads.get(algorithm); if (threads.isEmpty()) { return out; } out.reserve(threads.count() * 2); for (const auto &thread : threads.data()) { if (thread.index() >= devices.size()) { LOG_INFO("%s" YELLOW(" skip non-existing device with index ") YELLOW_BOLD("%u"), cuda_tag(), thread.index()); continue; } out.emplace_back(miner, algorithm, thread, devices[thread.index()]); } return out; } void xmrig::CudaConfig::read(const rapidjson::Value &value) { if (value.IsObject()) { m_enabled = Json::getBool(value, kEnabled, m_enabled); m_loader = Json::getString(value, kLoader); setDevicesHint(Json::getString(value, kDevicesHint)); m_threads.read(value); generate(); } else if (value.IsBool()) { m_enabled = value.GetBool(); generate(); } else { m_shouldSave = true; generate(); } } void xmrig::CudaConfig::generate() { if (generated) { return; } if (!isEnabled() || m_threads.has("*")) { return; } if (!CudaLib::init(loader())) { return; } if (!CudaLib::runtimeVersion() || !CudaLib::driverVersion() || !CudaLib::deviceCount()) { return; } const auto devices = CudaLib::devices(bfactor(), bsleep()); if (devices.empty()) { return; } size_t count = 0; count += xmrig::generate(m_threads, devices); count += xmrig::generate(m_threads, devices); count += xmrig::generate(m_threads, devices); count += xmrig::generate(m_threads, devices); count += xmrig::generate(m_threads, devices); generated = true; m_shouldSave = count > 0; } void xmrig::CudaConfig::setDevicesHint(const char *devicesHint) { if (devicesHint == nullptr) { return; } const auto indexes = String(devicesHint).split(','); m_devicesHint.reserve(indexes.size()); for (const auto &index : indexes) { m_devicesHint.push_back(strtoul(index, nullptr, 10)); } }