blob: ef44180cddba6fe842e687d276bceef39ed9b1f7 [file] [log] [blame]
Alexandru Gheorghec5463582018-03-27 15:52:02 +01001/*
2 * Copyright (C) 2018 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#define LOG_TAG "hwc-resource-manager"
18
Roman Stratiienko13cc3662020-08-29 21:35:39 +030019#include "ResourceManager.h"
Alexandru Gheorghec5463582018-03-27 15:52:02 +010020
Roman Stratiienkod21071f2021-03-09 21:56:50 +020021#include <fcntl.h>
Matvii Zorinec75ccd2020-07-17 12:08:45 +030022#include <sys/stat.h>
Roman Stratiienko13cc3662020-08-29 21:35:39 +030023
Alexandru Gheorghec5463582018-03-27 15:52:02 +010024#include <sstream>
Alexandru Gheorghec5463582018-03-27 15:52:02 +010025
Roman Stratiienkob2e9fe22020-10-03 10:52:36 +030026#include "bufferinfo/BufferInfoGetter.h"
Roman Stratiienkod518a052021-02-25 19:15:14 +020027#include "utils/log.h"
Roman Stratiienkod21071f2021-03-09 21:56:50 +020028#include "utils/properties.h"
Roman Stratiienkob2e9fe22020-10-03 10:52:36 +030029
Alexandru Gheorghec5463582018-03-27 15:52:02 +010030namespace android {
31
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020032ResourceManager::ResourceManager() : num_displays_(0), gralloc_(nullptr) {
Alexandru Gheorghec5463582018-03-27 15:52:02 +010033}
34
35int ResourceManager::Init() {
36 char path_pattern[PROPERTY_VALUE_MAX];
37 // Could be a valid path or it can have at the end of it the wildcard %
38 // which means that it will try open all devices until an error is met.
Jason Macnakf1af9572020-08-20 11:49:51 -070039 int path_len = property_get("vendor.hwc.drm.device", path_pattern,
40 "/dev/dri/card%");
Alexandru Gheorghec5463582018-03-27 15:52:02 +010041 int ret = 0;
42 if (path_pattern[path_len - 1] != '%') {
43 ret = AddDrmDevice(std::string(path_pattern));
44 } else {
45 path_pattern[path_len - 1] = '\0';
46 for (int idx = 0; !ret; ++idx) {
47 std::ostringstream path;
48 path << path_pattern << idx;
Matvii Zorinec75ccd2020-07-17 12:08:45 +030049
Roman Stratiienkob3b5c1e2021-02-15 13:44:19 +020050 struct stat buf {};
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020051 if (stat(path.str().c_str(), &buf))
Matvii Zorinec75ccd2020-07-17 12:08:45 +030052 break;
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020053
54 if (IsKMSDev(path.str().c_str()))
Matvii Zorinec75ccd2020-07-17 12:08:45 +030055 ret = AddDrmDevice(path.str());
Alexandru Gheorghec5463582018-03-27 15:52:02 +010056 }
57 }
58
59 if (!num_displays_) {
60 ALOGE("Failed to initialize any displays");
61 return ret ? -EINVAL : ret;
62 }
63
Roman Stratiienko65f2ba82019-12-20 17:04:01 +020064 char scale_with_gpu[PROPERTY_VALUE_MAX];
Jason Macnakf1af9572020-08-20 11:49:51 -070065 property_get("vendor.hwc.drm.scale_with_gpu", scale_with_gpu, "0");
Roman Stratiienko65f2ba82019-12-20 17:04:01 +020066 scale_with_gpu_ = bool(strncmp(scale_with_gpu, "0", 1));
67
Roman Stratiienkob2e9fe22020-10-03 10:52:36 +030068 if (!BufferInfoGetter::GetInstance()) {
69 ALOGE("Failed to initialize BufferInfoGetter");
70 return -EINVAL;
71 }
72
Alexandru Gheorghec5463582018-03-27 15:52:02 +010073 return hw_get_module(GRALLOC_HARDWARE_MODULE_ID,
74 (const hw_module_t **)&gralloc_);
75}
76
Roman Stratiienko8666dc92021-02-09 17:49:55 +020077int ResourceManager::AddDrmDevice(const std::string &path) {
78 auto drm = std::make_unique<DrmDevice>();
Roman Stratiienkob3b5c1e2021-02-15 13:44:19 +020079 int displays_added = 0;
80 int ret = 0;
Alexandru Gheorghec5463582018-03-27 15:52:02 +010081 std::tie(ret, displays_added) = drm->Init(path.c_str(), num_displays_);
Alexandru Gheorghec5463582018-03-27 15:52:02 +010082 drms_.push_back(std::move(drm));
83 num_displays_ += displays_added;
84 return ret;
85}
86
Alexandru Gheorgheb6a675e2018-03-27 16:10:55 +010087DrmConnector *ResourceManager::AvailableWritebackConnector(int display) {
88 DrmDevice *drm_device = GetDrmDevice(display);
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020089 DrmConnector *writeback_conn = nullptr;
Alexandru Gheorgheb6a675e2018-03-27 16:10:55 +010090 if (drm_device) {
91 writeback_conn = drm_device->AvailableWritebackConnector(display);
92 if (writeback_conn)
93 return writeback_conn;
94 }
95 for (auto &drm : drms_) {
96 if (drm.get() == drm_device)
97 continue;
98 writeback_conn = drm->AvailableWritebackConnector(display);
99 if (writeback_conn)
100 return writeback_conn;
101 }
102 return writeback_conn;
103}
104
Matvii Zorinec75ccd2020-07-17 12:08:45 +0300105bool ResourceManager::IsKMSDev(const char *path) {
106 int fd = open(path, O_RDWR | O_CLOEXEC);
107 if (fd < 0)
108 return false;
109
Roman Stratiienkod21071f2021-03-09 21:56:50 +0200110 auto *res = drmModeGetResources(fd);
Matvii Zorinec75ccd2020-07-17 12:08:45 +0300111 if (!res) {
112 close(fd);
113 return false;
114 }
115
116 bool is_kms = res->count_crtcs > 0 && res->count_connectors > 0 &&
117 res->count_encoders > 0;
118
119 drmModeFreeResources(res);
120 close(fd);
121
122 return is_kms;
123}
124
Alexandru Gheorghec5463582018-03-27 15:52:02 +0100125DrmDevice *ResourceManager::GetDrmDevice(int display) {
126 for (auto &drm : drms_) {
127 if (drm->HandlesDisplay(display))
128 return drm.get();
129 }
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +0200130 return nullptr;
Alexandru Gheorghec5463582018-03-27 15:52:02 +0100131}
132
Alexandru Gheorghec5463582018-03-27 15:52:02 +0100133const gralloc_module_t *ResourceManager::gralloc() {
134 return gralloc_;
135}
Sean Paulf72cccd2018-08-27 13:59:08 -0400136} // namespace android