diff options
Diffstat (limited to 'drm/DrmConnector.cpp')
-rw-r--r-- | drm/DrmConnector.cpp | 254 |
1 files changed, 254 insertions, 0 deletions
diff --git a/drm/DrmConnector.cpp b/drm/DrmConnector.cpp new file mode 100644 index 0000000..f1b6c1b --- /dev/null +++ b/drm/DrmConnector.cpp @@ -0,0 +1,254 @@ +/* + * 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 !blob; +} + +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 |