summaryrefslogtreecommitdiff
path: root/src/cache.cpp
blob: 6080f08294a72ac9a28e9826ce3d5ad640ea49bc (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
// =================================================================================================
// This file is part of the CLBlast project. The project is licensed under Apache Version 2.0. This
// project loosely follows the Google C++ styleguide and uses a tab-size of two spaces and a max-
// width of 100 characters per line.
//
// Author(s):
//   Cedric Nugteren <www.cedricnugteren.nl>
//
// This file implements the caching functionality of compiled binaries and programs.
//
// =================================================================================================

#include <string>
#include <vector>
#include <mutex>

#include "cache.hpp"

namespace clblast {
// =================================================================================================

// Stores the compiled binary or IR in the cache
void StoreBinaryToCache(const std::string &binary, const std::string &device_name,
                        const Precision &precision, const std::string &routine_name) {
  #ifdef VERBOSE
    printf("[DEBUG] Storing binary in cache\n");
  #endif
  binary_cache_mutex_.lock();
  binary_cache_.push_back(BinaryCache{binary, device_name, precision, routine_name});
  binary_cache_mutex_.unlock();
}

// Stores the compiled program in the cache
void StoreProgramToCache(const Program &program, const Context &context,
                         const Precision &precision, const std::string &routine_name) {
  #ifdef VERBOSE
    printf("[DEBUG] Storing program in cache\n");
  #endif
  program_cache_mutex_.lock();
  program_cache_.push_back(ProgramCache{program, context(), precision, routine_name});
  program_cache_mutex_.unlock();
}

// Queries the cache and retrieves a matching binary. Assumes that the match is available, throws
// otherwise.
const std::string& GetBinaryFromCache(const std::string &device_name, const Precision &precision,
                                      const std::string &routine_name) {
  #ifdef VERBOSE
    printf("[DEBUG] Retrieving binary from cache\n");
  #endif
  binary_cache_mutex_.lock();
  for (auto &cached_binary: binary_cache_) {
    if (cached_binary.MatchInCache(device_name, precision, routine_name)) {
      binary_cache_mutex_.unlock();
      return cached_binary.binary;
    }
  }
  binary_cache_mutex_.unlock();
  throw std::runtime_error("Internal CLBlast error: Expected binary in cache, but found none.");
}

// Queries the cache and retrieves a matching program. Assumes that the match is available, throws
// otherwise.
const Program& GetProgramFromCache(const Context &context, const Precision &precision,
                                   const std::string &routine_name) {
  #ifdef VERBOSE
    printf("[DEBUG] Retrieving program from cache\n");
  #endif
  program_cache_mutex_.lock();
  for (auto &cached_program: program_cache_) {
    if (cached_program.MatchInCache(context(), precision, routine_name)) {
      program_cache_mutex_.unlock();
      return cached_program.program;
    }
  }
  program_cache_mutex_.unlock();
  throw std::runtime_error("Internal CLBlast error: Expected program in cache, but found none.");
}

// Queries the cache to see whether or not the compiled kernel is already there
bool BinaryIsInCache(const std::string &device_name, const Precision &precision,
                     const std::string &routine_name) {
  binary_cache_mutex_.lock();
  for (auto &cached_binary: binary_cache_) {
    if (cached_binary.MatchInCache(device_name, precision, routine_name)) {
      binary_cache_mutex_.unlock();
      return true;
    }
  }
  binary_cache_mutex_.unlock();
  return false;
}

// Queries the cache to see whether or not the compiled kernel is already there
bool ProgramIsInCache(const Context &context, const Precision &precision,
                      const std::string &routine_name) {
  program_cache_mutex_.lock();
  for (auto &cached_program: program_cache_) {
    if (cached_program.MatchInCache(context(), precision, routine_name)) {
      program_cache_mutex_.unlock();
      return true;
    }
  }
  program_cache_mutex_.unlock();
  return false;
}

// =================================================================================================

// Clears the cache of stored binaries and programs
StatusCode CacheClearAll() {
  binary_cache_mutex_.lock();
  binary_cache_.clear();
  binary_cache_mutex_.unlock();
  program_cache_mutex_.lock();
  program_cache_.clear();
  program_cache_mutex_.unlock();
  return StatusCode::kSuccess;
}

// =================================================================================================
} // namespace clblast