summaryrefslogtreecommitdiff
path: root/libhwc2.1/libdevice/DisplayColorLoader.h
blob: 02bbbf8cd29c8067dec05179aed6a1408c40f6a1 (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
/*
 * Copyright (C) 2022 The Android Open Source Project
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

#ifndef DISPLAY_COLOR_LOADER_H
#define DISPLAY_COLOR_LOADER_H

#include <dlfcn.h>
#include <log/log.h>
#include <string>
#include <vector>

template <typename GsInterfaceType, const char *entry>
class DisplayColorLoader {
    static_assert(entry != nullptr);
    public:
      DisplayColorLoader(const DisplayColorLoader &) = delete;
      DisplayColorLoader& operator=(const DisplayColorLoader &) = delete;

      DisplayColorLoader(const std::string &libName) {
          libHandle = dlopen(libName.c_str(), RTLD_LAZY);

          if (libHandle == nullptr) {
              ALOGE("%s: failed to load library %s\n", __func__, libName.c_str());
              getDisplayColor = nullptr;
              return;
          }

          const displaycolor::DisplayColorIntfVer *(*getVersion)();
          getVersion = reinterpret_cast<decltype(getVersion)>(dlsym(libHandle, "GetInterfaceVersion"));
          if (getVersion == nullptr) {
              ALOGE("%s: prebuilt lib is not versioned", __func__);
              return;
          }

          auto intfVer = getVersion();

          if (intfVer != nullptr &&
              displaycolor::kInterfaceVersion.Compatible(*intfVer)) {
              getDisplayColor = reinterpret_cast<decltype(getDisplayColor)>(dlsym(libHandle, entry));

              if (getDisplayColor == nullptr) {
                  ALOGE("%s: failed to get %s\n", __func__, entry);
              } else if (!(displaycolor::kInterfaceVersion == *intfVer)) {
                  ALOGW("%s: different hwc/displaycolor patch level %u.%u.%u vs .%u",
                        __func__,
                        intfVer->major,
                        intfVer->minor,
                        displaycolor::kInterfaceVersion.patch,
                        intfVer->patch);
              }
          } else {
              if (intfVer != nullptr) {
                  ALOGE("%s: prebuilt lib version %u.%u.%u expected %u.%u.%u",
                        __func__,
                        intfVer->major,
                        intfVer->minor,
                        intfVer->patch,
                        displaycolor::kInterfaceVersion.major,
                        displaycolor::kInterfaceVersion.minor,
                        displaycolor::kInterfaceVersion.patch);
              } else {
                  ALOGE("%s: prebult lib getVersion returns null", __func__);
              }
          }
      }

      GsInterfaceType *GetDisplayColor(
              const std::vector<displaycolor::DisplayInfo> &display_info) {
          if (getDisplayColor != nullptr) {
              return getDisplayColor(display_info);
          }

          return nullptr;
      }

      ~DisplayColorLoader() {
          if (libHandle != nullptr) {
              dlclose(libHandle);
          }
      }

    private:
      void *libHandle;
      GsInterfaceType *(*getDisplayColor)(const std::vector<displaycolor::DisplayInfo> &);
};

#endif //DISPLAY_COLOR_LOADER_H