DPC++ Runtime
Runtime libraries for oneAPI DPC++
persistent_device_code_cache.hpp
Go to the documentation of this file.
1 //==---------- persistent_device_code_cache.hpp -----------------*- C++-*---==//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #pragma once
10 
11 #include <detail/config.hpp>
13 #include <fcntl.h>
14 #include <string>
15 #include <sycl/detail/os_util.hpp>
16 #include <sycl/detail/ur.hpp>
17 #include <sycl/detail/util.hpp>
18 #include <sycl/device.hpp>
19 #include <sys/stat.h>
20 #include <thread>
21 #include <vector>
22 
23 namespace sycl {
24 inline namespace _V1 {
25 namespace detail {
26 
27 /* This is temporary solution until std::filesystem is available when SYCL RT
28  * is moved to c++17 standard*/
29 std::string getDirName(const char *Path);
30 
31 /* The class manages inter-process synchronization:
32  * - Path passed to the constructor is appended with .lock and used as lock
33  * file.
34  * - All operations are not blocking and failure ignoring (diagnostic may be
35  * sent to std::cerr when SYCL_CACHE_TRACE environment variable is set).
36  * - There are two modes of accessing shared resource:
37  * - write access assumes that lock is acquired (object is created and
38  * isOwned() method confirms that current executor owns the lock);
39  * - read access checks that the lock is not acquired for write by others
40  * with the help of isLocked() method.
41  */
43 private:
44  const std::string FileName;
45  bool Owned = false;
46  static const char LockSuffix[];
47 
48 public:
49  LockCacheItem(const std::string &Path);
50 
51  bool isOwned() { return Owned; }
52  static bool isLocked(const std::string &Path) {
53  return OSUtil::isPathPresent(Path + LockSuffix);
54  }
56 };
57 /* End of temporary solution*/
58 
60  /* The device code images are stored on file system using structure below:
61  * <cache_root>/
62  * <device_hash>/
63  * <device_image_hash>/
64  * <spec_constants_values_hash>/
65  * <build_options_hash>/
66  * <n>.src
67  * <n>.bin
68  * .lock
69  * <cache_root> - root directory storing cache files;
70  * <device_hash> - hash out of device information used to
71  * identify target device;
72  * <device_image_hash> - hash made out of device images used as
73  * input for the JIT compilation;
74  * <spec_constants_values_hash> - hash for specialization constants values;
75  * <build_options_hash> - hash for all build options;
76  * <n> - sequential number of hash collisions.
77  * When hashes match for the specific build
78  * but full values don't, new cache item is
79  * added with incremented value(enumeration
80  * started from 0).
81  * Two files per cache item are stored on disk:
82  * <n>.src - contains full values for build parameters (device information,
83  * specialization constant values, build options, device images)
84  * which is used to resolve hash collisions and analysis of
85  * cached items.
86  * <n>.bin - contains built device code.
87  * <n>.lock - cache item lock file. It is created when data is saved to
88  * filesystem. On read operation the absence of file is checked
89  * but it is not created to avoid lock.
90  * All filesystem operation failures are not treated as SYCL errors and
91  * ignored. If such errors happen warning messages are written to std::cerr
92  * and:
93  * - on cache write operation cache item is not created;
94  * - on cache read operation it is treated as cache miss.
95  */
96 private:
97  /* Write built binary to persistent cache
98  * Format: numImages, 1stImageSize, Image[, NthImageSize, NthImage...]
99  */
100  static void writeBinaryDataToFile(const std::string &FileName,
101  const std::vector<std::vector<char>> &Data);
102 
103  /* Read built binary to persistent cache
104  * Format: numImages, 1stImageSize, Image[, NthImageSize, NthImage...]
105  */
106  static std::vector<std::vector<char>>
107  readBinaryDataFromFile(const std::string &FileName);
108 
109  /* Writing cache item key sources to be used for reliable identification
110  * Format: Four pairs of [size, value] for device, build options,
111  * specialization constant values, device code SPIR-V images.
112  */
113  static void
114  writeSourceItem(const std::string &FileName, const device &Device,
115  const std::vector<const RTDeviceBinaryImage *> &SortedImgs,
116  const SerializedObj &SpecConsts,
117  const std::string &BuildOptionsString);
118 
119  /* Check that cache item key sources are equal to the current program
120  */
121  static bool isCacheItemSrcEqual(
122  const std::string &FileName, const device &Device,
123  const std::vector<const RTDeviceBinaryImage *> &SortedImgs,
124  const SerializedObj &SpecConsts, const std::string &BuildOptionsString);
125 
126  /* Check if on-disk cache enabled.
127  */
128  static bool isEnabled();
129 
130  /* Returns the path to directory storing persistent device code cache.*/
131  static std::string getRootDir();
132 
133  /* Form string representing device version */
134  static std::string getDeviceIDString(const device &Device);
135 
136  /* Returns true if specified images should be cached on disk. It checks if
137  * cache is enabled, images have SPIRV type and match thresholds. */
138  static bool areImagesCacheable(
139  const std::vector<const RTDeviceBinaryImage *> &SortedImgs);
140 
141  /* Returns value of specified parameter. Default value is used if failure
142  * happens during obtaining value. */
143  template <ConfigID Config>
144  static unsigned long getNumParam(unsigned long Default) {
145  auto Value = SYCLConfig<Config>::get();
146  try {
147  if (Value)
148  return std::stol(Value);
149  } catch (std::exception const &) {
150  PersistentDeviceCodeCache::trace("Invalid value provided, use default " +
151  std::to_string(Default));
152  }
153  return Default;
154  }
155 
156  /* Default value for minimum device code size to be cached on disk in bytes */
157  static constexpr unsigned long DEFAULT_MIN_DEVICE_IMAGE_SIZE = 0;
158 
159  /* Default value for maximum device code size to be cached on disk in bytes */
160  static constexpr unsigned long DEFAULT_MAX_DEVICE_IMAGE_SIZE =
161  1024 * 1024 * 1024;
162 
163 public:
164  /* Get directory name for storing current cache item
165  */
166  static std::string
167  getCacheItemPath(const device &Device,
168  const std::vector<const RTDeviceBinaryImage *> &SortedImgs,
169  const SerializedObj &SpecConsts,
170  const std::string &BuildOptionsString);
171 
172  /* Program binaries built for one or more devices are read from persistent
173  * cache and returned in form of vector of programs. Each binary program is
174  * stored in vector of chars.
175  */
176  static std::vector<std::vector<char>>
177  getItemFromDisc(const device &Device,
178  const std::vector<const RTDeviceBinaryImage *> &Imgs,
179  const SerializedObj &SpecConsts,
180  const std::string &BuildOptionsString);
181 
182  /* Stores build program in persistent cache
183  */
184  static void
185  putItemToDisc(const device &Device,
186  const std::vector<const RTDeviceBinaryImage *> &Imgs,
187  const SerializedObj &SpecConsts,
188  const std::string &BuildOptionsString,
189  const ur_program_handle_t &NativePrg);
190 
191  /* Sends message to std:cerr stream when SYCL_CACHE_TRACE environemnt is set*/
192  static void trace(const std::string &msg) {
193  static const char *TraceEnabled = SYCLConfig<SYCL_CACHE_TRACE>::get();
194  if (TraceEnabled)
195  std::cerr << "*** Code caching: " << msg << std::endl;
196  }
197 };
198 } // namespace detail
199 } // namespace _V1
200 } // namespace sycl
static bool isLocked(const std::string &Path)
static bool isPathPresent(const std::string &Path)
Checks if specified path is present.
Definition: os_util.hpp:72
static std::string getCacheItemPath(const device &Device, const std::vector< const RTDeviceBinaryImage * > &SortedImgs, const SerializedObj &SpecConsts, const std::string &BuildOptionsString)
static std::vector< std::vector< char > > getItemFromDisc(const device &Device, const std::vector< const RTDeviceBinaryImage * > &Imgs, const SerializedObj &SpecConsts, const std::string &BuildOptionsString)
static void putItemToDisc(const device &Device, const std::vector< const RTDeviceBinaryImage * > &Imgs, const SerializedObj &SpecConsts, const std::string &BuildOptionsString, const ur_program_handle_t &NativePrg)
static const char * get()
Definition: config.hpp:115
The SYCL device class encapsulates a single SYCL device on which kernels may be executed.
Definition: device.hpp:64
__SYCL_EXTERN_STREAM_ATTRS ostream cerr
Linked to standard error (unbuffered)
std::string getDirName(const char *Path)
std::vector< unsigned char > SerializedObj
Definition: util.hpp:69
Definition: access.hpp:18
C++ utilities for Unified Runtime integration.