blob: 76a36e1d3ab54fc81d5d83ca57cd83542bdc1dba [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 Stratiienko123cdb02021-09-29 12:47:00 +030032ResourceManager::ResourceManager() : num_displays_(0) {
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
Roman Stratiienko123cdb02021-09-29 12:47:00 +030073 return 0;
Alexandru Gheorghec5463582018-03-27 15:52:02 +010074}
75
Roman Stratiienko8666dc92021-02-09 17:49:55 +020076int ResourceManager::AddDrmDevice(const std::string &path) {
77 auto drm = std::make_unique<DrmDevice>();
Roman Stratiienkob3b5c1e2021-02-15 13:44:19 +020078 int displays_added = 0;
79 int ret = 0;
Alexandru Gheorghec5463582018-03-27 15:52:02 +010080 std::tie(ret, displays_added) = drm->Init(path.c_str(), num_displays_);
Alexandru Gheorghec5463582018-03-27 15:52:02 +010081 drms_.push_back(std::move(drm));
82 num_displays_ += displays_added;
83 return ret;
84}
85
Alexandru Gheorgheb6a675e2018-03-27 16:10:55 +010086DrmConnector *ResourceManager::AvailableWritebackConnector(int display) {
87 DrmDevice *drm_device = GetDrmDevice(display);
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +020088 DrmConnector *writeback_conn = nullptr;
Alexandru Gheorgheb6a675e2018-03-27 16:10:55 +010089 if (drm_device) {
90 writeback_conn = drm_device->AvailableWritebackConnector(display);
91 if (writeback_conn)
92 return writeback_conn;
93 }
94 for (auto &drm : drms_) {
95 if (drm.get() == drm_device)
96 continue;
97 writeback_conn = drm->AvailableWritebackConnector(display);
98 if (writeback_conn)
99 return writeback_conn;
100 }
101 return writeback_conn;
102}
103
Matvii Zorinec75ccd2020-07-17 12:08:45 +0300104bool ResourceManager::IsKMSDev(const char *path) {
Roman Stratiienko3e8ce572021-09-29 12:46:28 +0300105 auto fd = UniqueFd(open(path, O_RDWR | O_CLOEXEC));
106 if (!fd) {
Matvii Zorinec75ccd2020-07-17 12:08:45 +0300107 return false;
Roman Stratiienko3e8ce572021-09-29 12:46:28 +0300108 }
Matvii Zorinec75ccd2020-07-17 12:08:45 +0300109
Roman Stratiienko3e8ce572021-09-29 12:46:28 +0300110 auto res = MakeDrmModeResUnique(fd.Get());
Matvii Zorinec75ccd2020-07-17 12:08:45 +0300111 if (!res) {
Matvii Zorinec75ccd2020-07-17 12:08:45 +0300112 return false;
113 }
114
115 bool is_kms = res->count_crtcs > 0 && res->count_connectors > 0 &&
116 res->count_encoders > 0;
117
Matvii Zorinec75ccd2020-07-17 12:08:45 +0300118 return is_kms;
119}
120
Alexandru Gheorghec5463582018-03-27 15:52:02 +0100121DrmDevice *ResourceManager::GetDrmDevice(int display) {
122 for (auto &drm : drms_) {
123 if (drm->HandlesDisplay(display))
124 return drm.get();
125 }
Roman Stratiienkoe2f2c922021-02-13 10:57:47 +0200126 return nullptr;
Alexandru Gheorghec5463582018-03-27 15:52:02 +0100127}
Sean Paulf72cccd2018-08-27 13:59:08 -0400128} // namespace android