system.hpp 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  1. //---------------------------------------------------------------------------//
  2. // Copyright (c) 2013 Kyle Lutz <kyle.r.lutz@gmail.com>
  3. //
  4. // Distributed under the Boost Software License, Version 1.0
  5. // See accompanying file LICENSE_1_0.txt or copy at
  6. // http://www.boost.org/LICENSE_1_0.txt
  7. //
  8. // See http://boostorg.github.com/compute for more information.
  9. //---------------------------------------------------------------------------//
  10. #ifndef BOOST_COMPUTE_SYSTEM_HPP
  11. #define BOOST_COMPUTE_SYSTEM_HPP
  12. #include <string>
  13. #include <vector>
  14. #include <cstdlib>
  15. #include <boost/throw_exception.hpp>
  16. #include <boost/compute/cl.hpp>
  17. #include <boost/compute/device.hpp>
  18. #include <boost/compute/context.hpp>
  19. #include <boost/compute/platform.hpp>
  20. #include <boost/compute/command_queue.hpp>
  21. #include <boost/compute/detail/getenv.hpp>
  22. #include <boost/compute/exception/no_device_found.hpp>
  23. namespace boost {
  24. namespace compute {
  25. /// \class system
  26. /// \brief Provides access to platforms and devices on the system.
  27. ///
  28. /// The system class contains a set of static functions which provide access to
  29. /// the OpenCL platforms and compute devices on the host system.
  30. ///
  31. /// The default_device() convenience method automatically selects and returns
  32. /// the "best" compute device for the system following a set of heuristics and
  33. /// environment variables. This simplifies setup of the OpenCL enviornment.
  34. ///
  35. /// \see platform, device, context
  36. class system
  37. {
  38. public:
  39. /// Returns the default compute device for the system.
  40. ///
  41. /// The default device is selected based on a set of heuristics and can be
  42. /// influenced using one of the following environment variables:
  43. ///
  44. /// \li \c BOOST_COMPUTE_DEFAULT_DEVICE -
  45. /// name of the compute device (e.g. "GTX TITAN")
  46. /// \li \c BOOST_COMPUTE_DEFAULT_DEVICE_TYPE
  47. /// type of the compute device (e.g. "GPU" or "CPU")
  48. /// \li \c BOOST_COMPUTE_DEFAULT_PLATFORM -
  49. /// name of the platform (e.g. "NVIDIA CUDA")
  50. /// \li \c BOOST_COMPUTE_DEFAULT_VENDOR -
  51. /// name of the device vendor (e.g. "NVIDIA")
  52. ///
  53. /// The default device is determined once on the first time this function
  54. /// is called. Calling this function multiple times will always result in
  55. /// the same device being returned.
  56. ///
  57. /// If no OpenCL device is found on the system, a no_device_found exception
  58. /// is thrown.
  59. ///
  60. /// For example, to print the name of the default compute device on the
  61. /// system:
  62. /// \code
  63. /// // get the default compute device
  64. /// boost::compute::device device = boost::compute::system::default_device();
  65. ///
  66. /// // print the name of the device
  67. /// std::cout << "default device: " << device.name() << std::endl;
  68. /// \endcode
  69. static device default_device()
  70. {
  71. static device default_device = find_default_device();
  72. return default_device;
  73. }
  74. /// Returns the device with \p name.
  75. ///
  76. /// \throws no_device_found if no device with \p name is found.
  77. static device find_device(const std::string &name)
  78. {
  79. const std::vector<device> devices = system::devices();
  80. for(size_t i = 0; i < devices.size(); i++){
  81. const device& device = devices[i];
  82. if(device.name() == name){
  83. return device;
  84. }
  85. }
  86. BOOST_THROW_EXCEPTION(no_device_found());
  87. }
  88. /// Returns a vector containing all of the compute devices on
  89. /// the system.
  90. ///
  91. /// For example, to print out the name of each OpenCL-capable device
  92. /// available on the system:
  93. /// \code
  94. /// for(const auto &device : boost::compute::system::devices()){
  95. /// std::cout << device.name() << std::endl;
  96. /// }
  97. /// \endcode
  98. static std::vector<device> devices()
  99. {
  100. std::vector<device> devices;
  101. const std::vector<platform> platforms = system::platforms();
  102. for(size_t i = 0; i < platforms.size(); i++){
  103. const std::vector<device> platform_devices = platforms[i].devices();
  104. devices.insert(
  105. devices.end(), platform_devices.begin(), platform_devices.end()
  106. );
  107. }
  108. return devices;
  109. }
  110. /// Returns the number of compute devices on the system.
  111. static size_t device_count()
  112. {
  113. size_t count = 0;
  114. const std::vector<platform> platforms = system::platforms();
  115. for(size_t i = 0; i < platforms.size(); i++){
  116. count += platforms[i].device_count();
  117. }
  118. return count;
  119. }
  120. /// Returns the default context for the system.
  121. ///
  122. /// The default context is created for the default device on the system
  123. /// (as returned by default_device()).
  124. ///
  125. /// The default context is created once on the first time this function is
  126. /// called. Calling this function multiple times will always result in the
  127. /// same context object being returned.
  128. static context default_context()
  129. {
  130. static context default_context(default_device());
  131. return default_context;
  132. }
  133. /// Returns the default command queue for the system.
  134. static command_queue& default_queue()
  135. {
  136. static command_queue queue(default_context(), default_device());
  137. return queue;
  138. }
  139. /// Blocks until all outstanding computations on the default
  140. /// command queue are complete.
  141. ///
  142. /// This is equivalent to:
  143. /// \code
  144. /// system::default_queue().finish();
  145. /// \endcode
  146. static void finish()
  147. {
  148. default_queue().finish();
  149. }
  150. /// Returns a vector containing each of the OpenCL platforms on the system.
  151. ///
  152. /// For example, to print out the name of each OpenCL platform present on
  153. /// the system:
  154. /// \code
  155. /// for(const auto &platform : boost::compute::system::platforms()){
  156. /// std::cout << platform.name() << std::endl;
  157. /// }
  158. /// \endcode
  159. static std::vector<platform> platforms()
  160. {
  161. cl_uint count = 0;
  162. clGetPlatformIDs(0, 0, &count);
  163. std::vector<cl_platform_id> platform_ids(count);
  164. clGetPlatformIDs(count, &platform_ids[0], 0);
  165. std::vector<platform> platforms;
  166. for(size_t i = 0; i < platform_ids.size(); i++){
  167. platforms.push_back(platform(platform_ids[i]));
  168. }
  169. return platforms;
  170. }
  171. /// Returns the number of compute platforms on the system.
  172. static size_t platform_count()
  173. {
  174. cl_uint count = 0;
  175. clGetPlatformIDs(0, 0, &count);
  176. return static_cast<size_t>(count);
  177. }
  178. private:
  179. /// \internal_
  180. static device find_default_device()
  181. {
  182. // get a list of all devices on the system
  183. const std::vector<device> devices_ = devices();
  184. if(devices_.empty()){
  185. BOOST_THROW_EXCEPTION(no_device_found());
  186. }
  187. // check for device from environment variable
  188. const char *name = detail::getenv("BOOST_COMPUTE_DEFAULT_DEVICE");
  189. const char *type = detail::getenv("BOOST_COMPUTE_DEFAULT_DEVICE_TYPE");
  190. const char *platform = detail::getenv("BOOST_COMPUTE_DEFAULT_PLATFORM");
  191. const char *vendor = detail::getenv("BOOST_COMPUTE_DEFAULT_VENDOR");
  192. if(name || type || platform || vendor){
  193. for(size_t i = 0; i < devices_.size(); i++){
  194. const device& device = devices_[i];
  195. if (name && !matches(device.name(), name))
  196. continue;
  197. if (type && matches(std::string("GPU"), type))
  198. if (!(device.type() & device::gpu))
  199. continue;
  200. if (type && matches(std::string("CPU"), type))
  201. if (!(device.type() & device::cpu))
  202. continue;
  203. if (platform && !matches(device.platform().name(), platform))
  204. continue;
  205. if (vendor && !matches(device.vendor(), vendor))
  206. continue;
  207. return device;
  208. }
  209. }
  210. // find the first gpu device
  211. for(size_t i = 0; i < devices_.size(); i++){
  212. const device& device = devices_[i];
  213. if(device.type() & device::gpu){
  214. return device;
  215. }
  216. }
  217. // find the first cpu device
  218. for(size_t i = 0; i < devices_.size(); i++){
  219. const device& device = devices_[i];
  220. if(device.type() & device::cpu){
  221. return device;
  222. }
  223. }
  224. // return the first device found
  225. return devices_[0];
  226. }
  227. /// \internal_
  228. static bool matches(const std::string &str, const std::string &pattern)
  229. {
  230. return str.find(pattern) != std::string::npos;
  231. }
  232. };
  233. } // end compute namespace
  234. } // end boost namespace
  235. #endif // BOOST_COMPUTE_SYSTEM_HPP