| /* |
| * Copyright (C) 2015 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. |
| */ |
| |
| #define LOG_TAG "hwc-drm-connector" |
| |
| #include "DrmConnector.h" |
| |
| #include <errno.h> |
| #include <log/log.h> |
| #include <stdint.h> |
| #include <xf86drmMode.h> |
| |
| #include <array> |
| #include <sstream> |
| |
| #include "DrmDevice.h" |
| |
| namespace android { |
| |
| constexpr size_t TYPES_COUNT = 17; |
| |
| DrmConnector::DrmConnector(DrmDevice *drm, drmModeConnectorPtr c, |
| DrmEncoder *current_encoder, |
| std::vector<DrmEncoder *> &possible_encoders) |
| : drm_(drm), |
| id_(c->connector_id), |
| encoder_(current_encoder), |
| display_(-1), |
| type_(c->connector_type), |
| type_id_(c->connector_type_id), |
| state_(c->connection), |
| mm_width_(c->mmWidth), |
| mm_height_(c->mmHeight), |
| possible_encoders_(possible_encoders) { |
| } |
| |
| int DrmConnector::Init() { |
| int ret = drm_->GetConnectorProperty(*this, "DPMS", &dpms_property_); |
| if (ret) { |
| ALOGE("Could not get DPMS property\n"); |
| return ret; |
| } |
| ret = drm_->GetConnectorProperty(*this, "CRTC_ID", &crtc_id_property_); |
| if (ret) { |
| ALOGE("Could not get CRTC_ID property\n"); |
| return ret; |
| } |
| ret = UpdateEdidProperty(); |
| if (writeback()) { |
| ret = drm_->GetConnectorProperty(*this, "WRITEBACK_PIXEL_FORMATS", |
| &writeback_pixel_formats_); |
| if (ret) { |
| ALOGE("Could not get WRITEBACK_PIXEL_FORMATS connector_id = %d\n", id_); |
| return ret; |
| } |
| ret = drm_->GetConnectorProperty(*this, "WRITEBACK_FB_ID", |
| &writeback_fb_id_); |
| if (ret) { |
| ALOGE("Could not get WRITEBACK_FB_ID connector_id = %d\n", id_); |
| return ret; |
| } |
| ret = drm_->GetConnectorProperty(*this, "WRITEBACK_OUT_FENCE_PTR", |
| &writeback_out_fence_); |
| if (ret) { |
| ALOGE("Could not get WRITEBACK_OUT_FENCE_PTR connector_id = %d\n", id_); |
| return ret; |
| } |
| } |
| return 0; |
| } |
| |
| int DrmConnector::UpdateEdidProperty() { |
| int ret = drm_->GetConnectorProperty(*this, "EDID", &edid_property_); |
| if (ret) { |
| ALOGW("Could not get EDID property\n"); |
| } |
| return ret; |
| } |
| |
| int DrmConnector::GetEdidBlob(drmModePropertyBlobPtr &blob) { |
| uint64_t blob_id; |
| int ret = UpdateEdidProperty(); |
| if (ret) { |
| return ret; |
| } |
| |
| std::tie(ret, blob_id) = edid_property().value(); |
| if (ret) { |
| return ret; |
| } |
| |
| blob = drmModeGetPropertyBlob(drm_->fd(), blob_id); |
| return 0; |
| } |
| |
| uint32_t DrmConnector::id() const { |
| return id_; |
| } |
| |
| int DrmConnector::display() const { |
| return display_; |
| } |
| |
| void DrmConnector::set_display(int display) { |
| display_ = display; |
| } |
| |
| bool DrmConnector::internal() const { |
| return type_ == DRM_MODE_CONNECTOR_LVDS || type_ == DRM_MODE_CONNECTOR_eDP || |
| type_ == DRM_MODE_CONNECTOR_DSI || |
| type_ == DRM_MODE_CONNECTOR_VIRTUAL || type_ == DRM_MODE_CONNECTOR_DPI; |
| } |
| |
| bool DrmConnector::external() const { |
| return type_ == DRM_MODE_CONNECTOR_HDMIA || |
| type_ == DRM_MODE_CONNECTOR_DisplayPort || |
| type_ == DRM_MODE_CONNECTOR_DVID || type_ == DRM_MODE_CONNECTOR_DVII || |
| type_ == DRM_MODE_CONNECTOR_VGA; |
| } |
| |
| bool DrmConnector::writeback() const { |
| #ifdef DRM_MODE_CONNECTOR_WRITEBACK |
| return type_ == DRM_MODE_CONNECTOR_WRITEBACK; |
| #else |
| return false; |
| #endif |
| } |
| |
| bool DrmConnector::valid_type() const { |
| return internal() || external() || writeback(); |
| } |
| |
| std::string DrmConnector::name() const { |
| constexpr std::array<const char *, TYPES_COUNT> names = |
| {"None", "VGA", "DVI-I", "DVI-D", "DVI-A", "Composite", |
| "SVIDEO", "LVDS", "Component", "DIN", "DP", "HDMI-A", |
| "HDMI-B", "TV", "eDP", "Virtual", "DSI"}; |
| |
| if (type_ < TYPES_COUNT) { |
| std::ostringstream name_buf; |
| name_buf << names[type_] << "-" << type_id_; |
| return name_buf.str(); |
| } else { |
| ALOGE("Unknown type in connector %d, could not make his name", id_); |
| return "None"; |
| } |
| } |
| |
| int DrmConnector::UpdateModes() { |
| int fd = drm_->fd(); |
| |
| drmModeConnectorPtr c = drmModeGetConnector(fd, id_); |
| if (!c) { |
| ALOGE("Failed to get connector %d", id_); |
| return -ENODEV; |
| } |
| |
| state_ = c->connection; |
| |
| bool preferred_mode_found = false; |
| std::vector<DrmMode> new_modes; |
| for (int i = 0; i < c->count_modes; ++i) { |
| bool exists = false; |
| for (const DrmMode &mode : modes_) { |
| if (mode == c->modes[i]) { |
| new_modes.push_back(mode); |
| exists = true; |
| break; |
| } |
| } |
| if (!exists) { |
| DrmMode m(&c->modes[i]); |
| m.set_id(drm_->next_mode_id()); |
| new_modes.push_back(m); |
| } |
| // Use only the first DRM_MODE_TYPE_PREFERRED mode found |
| if (!preferred_mode_found && |
| (new_modes.back().type() & DRM_MODE_TYPE_PREFERRED)) { |
| preferred_mode_id_ = new_modes.back().id(); |
| preferred_mode_found = true; |
| } |
| } |
| modes_.swap(new_modes); |
| if (!preferred_mode_found && modes_.size() != 0) { |
| preferred_mode_id_ = modes_[0].id(); |
| } |
| return 0; |
| } |
| |
| const DrmMode &DrmConnector::active_mode() const { |
| return active_mode_; |
| } |
| |
| void DrmConnector::set_active_mode(const DrmMode &mode) { |
| active_mode_ = mode; |
| } |
| |
| const DrmProperty &DrmConnector::dpms_property() const { |
| return dpms_property_; |
| } |
| |
| const DrmProperty &DrmConnector::crtc_id_property() const { |
| return crtc_id_property_; |
| } |
| |
| const DrmProperty &DrmConnector::edid_property() const { |
| return edid_property_; |
| } |
| |
| const DrmProperty &DrmConnector::writeback_pixel_formats() const { |
| return writeback_pixel_formats_; |
| } |
| |
| const DrmProperty &DrmConnector::writeback_fb_id() const { |
| return writeback_fb_id_; |
| } |
| |
| const DrmProperty &DrmConnector::writeback_out_fence() const { |
| return writeback_out_fence_; |
| } |
| |
| DrmEncoder *DrmConnector::encoder() const { |
| return encoder_; |
| } |
| |
| void DrmConnector::set_encoder(DrmEncoder *encoder) { |
| encoder_ = encoder; |
| } |
| |
| drmModeConnection DrmConnector::state() const { |
| return state_; |
| } |
| |
| uint32_t DrmConnector::mm_width() const { |
| return mm_width_; |
| } |
| |
| uint32_t DrmConnector::mm_height() const { |
| return mm_height_; |
| } |
| } // namespace android |