cf12defd28
Ultraworked with [Sisyphus](https://github.com/code-yeongyu/oh-my-openagent) Co-authored-by: Sisyphus <clio-agent@sisyphuslabs.ai>
404 lines
13 KiB
C++
404 lines
13 KiB
C++
/*
|
|
KWin - the KDE window manager
|
|
This file is part of the KDE project.
|
|
|
|
SPDX-FileCopyrightText: 2015 Martin Flöser <mgraesslin@kde.org>
|
|
SPDX-FileCopyrightText: 2019 Vlad Zahorodnii <vlad.zahorodnii@kde.org>
|
|
|
|
SPDX-License-Identifier: GPL-2.0-or-later
|
|
*/
|
|
#include "edid.h"
|
|
|
|
#include "config-kwin.h"
|
|
|
|
#include "c_ptr.h"
|
|
#include "common.h"
|
|
|
|
#include <QFile>
|
|
#include <QStandardPaths>
|
|
#include <cstdlib>
|
|
|
|
#include <KLocalizedString>
|
|
#include <QCryptographicHash>
|
|
|
|
extern "C" {
|
|
#include <libdisplay-info/cta.h>
|
|
#include <libdisplay-info/displayid.h>
|
|
#include <libdisplay-info/edid.h>
|
|
#include <libdisplay-info/info.h>
|
|
}
|
|
|
|
namespace KWin
|
|
{
|
|
|
|
static QByteArray parsePnpId(const uint8_t *data)
|
|
{
|
|
// Decode PNP ID from three 5 bit words packed into 2 bytes:
|
|
//
|
|
// | Byte | Bit |
|
|
// | | 7 | 6 | 5 | 4 | 3 | 2 | 1 | 0 |
|
|
// ----------------------------------------
|
|
// | 1 | 0)| (4| 3 | 2 | 1 | 0)| (4| 3 |
|
|
// | | * | Character 1 | Char 2|
|
|
// ----------------------------------------
|
|
// | 2 | 2 | 1 | 0)| (4| 3 | 2 | 1 | 0)|
|
|
// | | Character2| Character 3 |
|
|
// ----------------------------------------
|
|
const uint offset = 0x8;
|
|
|
|
char pnpId[4];
|
|
pnpId[0] = 'A' + ((data[offset + 0] >> 2) & 0x1f) - 1;
|
|
pnpId[1] = 'A' + (((data[offset + 0] & 0x3) << 3) | ((data[offset + 1] >> 5) & 0x7)) - 1;
|
|
pnpId[2] = 'A' + (data[offset + 1] & 0x1f) - 1;
|
|
pnpId[3] = '\0';
|
|
|
|
return QByteArray(pnpId);
|
|
}
|
|
|
|
static QByteArray parseEisaId(const uint8_t *data)
|
|
{
|
|
for (int i = 72; i <= 108; i += 18) {
|
|
// Skip the block if it isn't used as monitor descriptor.
|
|
if (data[i]) {
|
|
continue;
|
|
}
|
|
if (data[i + 1]) {
|
|
continue;
|
|
}
|
|
|
|
// We have found the EISA ID, it's stored as ASCII.
|
|
if (data[i + 3] == 0xfe) {
|
|
return QByteArray(reinterpret_cast<const char *>(&data[i + 5]), 13).trimmed();
|
|
}
|
|
}
|
|
|
|
// If there isn't an ASCII EISA ID descriptor, try to decode PNP ID
|
|
return parsePnpId(data);
|
|
}
|
|
|
|
static QByteArray parseVendor(const uint8_t *data)
|
|
{
|
|
const auto pnpId = parsePnpId(data);
|
|
|
|
// Map to vendor name
|
|
QFile pnpFile(QStandardPaths::locate(QStandardPaths::GenericDataLocation, QStringLiteral("hwdata/pnp.ids")));
|
|
if (pnpFile.exists() && pnpFile.open(QIODevice::ReadOnly)) {
|
|
while (!pnpFile.atEnd()) {
|
|
const auto line = pnpFile.readLine();
|
|
if (line.startsWith(pnpId)) {
|
|
return line.mid(4).trimmed();
|
|
}
|
|
}
|
|
}
|
|
|
|
return {};
|
|
}
|
|
|
|
static QSize determineScreenPhysicalSizeMm(const di_edid *edid)
|
|
{
|
|
// An EDID can contain zero or more detailed timing definitions, which can
|
|
// contain more precise physical dimensions (in millimeters, as opposed to
|
|
// centimeters). Pick the first sane physical dimension from detailed timings
|
|
// and fall back to the basic dimensions.
|
|
const struct di_edid_detailed_timing_def *const *detailedTimings = di_edid_get_detailed_timing_defs(edid);
|
|
// detailedTimings is a null-terminated array.
|
|
for (int i = 0; detailedTimings[i] != nullptr; i++) {
|
|
const struct di_edid_detailed_timing_def *timing = detailedTimings[i];
|
|
// Sanity check dimensions: physical aspect ratio should roughly equal
|
|
// mode aspect ratio (i.e. width_in_pixels / height_in_pixels).
|
|
// This assumes that the display has square pixels, but this is true for
|
|
// basically all modern displays.
|
|
if (timing->horiz_image_mm > 0 && timing->vert_image_mm > 0
|
|
&& timing->horiz_video > 0 && timing->vert_video > 0) {
|
|
const double physicalAspectRatio = double(timing->horiz_image_mm) / double(timing->vert_image_mm);
|
|
const double modeAspectRatio = double(timing->horiz_video) / double(timing->vert_video);
|
|
|
|
if (std::abs(physicalAspectRatio - modeAspectRatio) <= 0.1) {
|
|
return QSize(timing->horiz_image_mm, timing->vert_image_mm);
|
|
}
|
|
}
|
|
}
|
|
const di_edid_screen_size *screenSize = di_edid_get_screen_size(edid);
|
|
return QSize(screenSize->width_cm, screenSize->height_cm) * 10;
|
|
}
|
|
|
|
Edid::Edid()
|
|
{
|
|
}
|
|
|
|
Edid::Edid(const void *data, uint32_t size)
|
|
: Edid(QByteArrayView(reinterpret_cast<const uint8_t *>(data), size))
|
|
{
|
|
}
|
|
|
|
Edid::Edid(QByteArrayView data, std::optional<QByteArrayView> identifierOverride)
|
|
: Edid(data)
|
|
{
|
|
if (identifierOverride.has_value()) {
|
|
m_identifier = identifierOverride->toByteArray();
|
|
}
|
|
}
|
|
|
|
static const auto s_forceHdrSupport = []() -> std::optional<bool> {
|
|
bool ok = false;
|
|
int ret = qEnvironmentVariableIntValue("KWIN_FORCE_ASSUME_HDR_SUPPORT", &ok);
|
|
if (ok) {
|
|
return ret == 1;
|
|
} else {
|
|
return std::nullopt;
|
|
}
|
|
}();
|
|
|
|
Edid::Edid(QByteArrayView data)
|
|
{
|
|
m_raw = QByteArray(data.data(), data.size());
|
|
if (m_raw.isEmpty()) {
|
|
return;
|
|
}
|
|
|
|
QCryptographicHash hash(QCryptographicHash::Md5);
|
|
hash.addData(m_raw);
|
|
m_hash = QString::fromLatin1(hash.result().toHex());
|
|
|
|
auto info = di_info_parse_edid(data.data(), data.size());
|
|
if (!info) {
|
|
qCWarning(KWIN_CORE, "parsing edid failed");
|
|
return;
|
|
}
|
|
const di_edid *edid = di_info_get_edid(info);
|
|
const di_edid_vendor_product *productInfo = di_edid_get_vendor_product(edid);
|
|
|
|
const uint8_t *bytes = reinterpret_cast<const uint8_t *>(data.data());
|
|
|
|
// basic output information
|
|
m_physicalSize = determineScreenPhysicalSizeMm(edid);
|
|
m_eisaId = parseEisaId(bytes);
|
|
UniqueCPtr<char> monitorName{di_info_get_model(info)};
|
|
m_monitorName = QByteArray(monitorName.get());
|
|
UniqueCPtr<char> serial{di_info_get_serial(info)};
|
|
m_serialNumber = QByteArray(serial.get());
|
|
m_vendor = parseVendor(bytes);
|
|
|
|
m_identifier = QByteArray(productInfo->manufacturer, 3) + " " + QByteArray::number(productInfo->product) + " " + QByteArray::number(productInfo->serial) + " "
|
|
+ QByteArray::number(productInfo->manufacture_week) + " " + QByteArray::number(productInfo->manufacture_year) + " " + QByteArray::number(productInfo->model_year);
|
|
|
|
// colorimetry and HDR metadata
|
|
const auto chromaticity = di_edid_get_chromaticity_coords(edid);
|
|
if (chromaticity) {
|
|
const xy red{chromaticity->red_x, chromaticity->red_y};
|
|
const xy green{chromaticity->green_x, chromaticity->green_y};
|
|
const xy blue{chromaticity->blue_x, chromaticity->blue_y};
|
|
const xy white{chromaticity->white_x, chromaticity->white_y};
|
|
if (Colorimetry::isReal(red, green, blue, white)) {
|
|
m_colorimetry = Colorimetry{
|
|
red,
|
|
green,
|
|
blue,
|
|
white,
|
|
};
|
|
} else {
|
|
qCWarning(KWIN_CORE) << "EDID colorimetry" << red << green << blue << white << "is is invalid";
|
|
}
|
|
} else {
|
|
m_colorimetry.reset();
|
|
}
|
|
|
|
const di_edid_cta *cta = nullptr;
|
|
const di_displayid *displayid = nullptr;
|
|
const di_edid_ext *const *exts = di_edid_get_extensions(edid);
|
|
const di_cta_hdr_static_metadata_block *hdr_static_metadata = nullptr;
|
|
const di_cta_colorimetry_block *colorimetry = nullptr;
|
|
for (; *exts != nullptr; exts++) {
|
|
if (!cta && (cta = di_edid_ext_get_cta(*exts))) {
|
|
continue;
|
|
}
|
|
if (!displayid && (displayid = di_edid_ext_get_displayid(*exts))) {
|
|
continue;
|
|
}
|
|
}
|
|
if (cta) {
|
|
const di_cta_data_block *const *blocks = di_edid_cta_get_data_blocks(cta);
|
|
for (; *blocks != nullptr; blocks++) {
|
|
if (!hdr_static_metadata && (hdr_static_metadata = di_cta_data_block_get_hdr_static_metadata(*blocks))) {
|
|
continue;
|
|
}
|
|
if (!colorimetry && (colorimetry = di_cta_data_block_get_colorimetry(*blocks))) {
|
|
continue;
|
|
}
|
|
}
|
|
if (hdr_static_metadata) {
|
|
m_hdrMetadata = HDRMetadata{
|
|
.desiredContentMinLuminance = hdr_static_metadata->desired_content_min_luminance,
|
|
.desiredContentMaxLuminance = hdr_static_metadata->desired_content_max_luminance > 0 ? std::make_optional(hdr_static_metadata->desired_content_max_luminance) : std::nullopt,
|
|
.desiredMaxFrameAverageLuminance = hdr_static_metadata->desired_content_max_frame_avg_luminance > 0 ? std::make_optional(hdr_static_metadata->desired_content_max_frame_avg_luminance) : std::nullopt,
|
|
.supportsPQ = hdr_static_metadata->eotfs->pq,
|
|
.supportsBT2020 = colorimetry && colorimetry->bt2020_rgb,
|
|
};
|
|
}
|
|
}
|
|
if (s_forceHdrSupport.has_value()) {
|
|
if (!m_hdrMetadata) {
|
|
m_hdrMetadata = HDRMetadata{
|
|
.desiredContentMinLuminance = 0,
|
|
.desiredContentMaxLuminance = std::nullopt,
|
|
.desiredMaxFrameAverageLuminance = std::nullopt,
|
|
.supportsPQ = *s_forceHdrSupport,
|
|
.supportsBT2020 = *s_forceHdrSupport,
|
|
};
|
|
} else {
|
|
m_hdrMetadata->supportsPQ = *s_forceHdrSupport;
|
|
m_hdrMetadata->supportsBT2020 = *s_forceHdrSupport;
|
|
}
|
|
}
|
|
if (displayid) {
|
|
const di_displayid_display_params *params = nullptr;
|
|
const di_displayid_type_i_ii_vii_timing *const *type1Timings = nullptr;
|
|
const di_displayid_type_i_ii_vii_timing *const *type2Timings = nullptr;
|
|
for (auto block = di_displayid_get_data_blocks(displayid); *block != nullptr; block++) {
|
|
if (!params && (params = di_displayid_data_block_get_display_params(*block))) {
|
|
continue;
|
|
}
|
|
if (!type1Timings && (type1Timings = di_displayid_data_block_get_type_i_timings(*block))) {
|
|
continue;
|
|
}
|
|
if (!type2Timings && (type2Timings = di_displayid_data_block_get_type_ii_timings(*block))) {
|
|
continue;
|
|
}
|
|
}
|
|
if (params && params->horiz_pixels != 0 && params->vert_pixels != 0) {
|
|
m_nativeResolution = QSize(params->horiz_pixels, params->vert_pixels);
|
|
}
|
|
if (type1Timings && !m_nativeResolution) {
|
|
for (auto timing = type1Timings; *timing != nullptr; timing++) {
|
|
if ((*timing)->preferred && (!m_nativeResolution || m_nativeResolution->width() < (*timing)->horiz_active || m_nativeResolution->height() < (*timing)->vert_active)) {
|
|
m_nativeResolution = QSize((*timing)->horiz_active, (*timing)->vert_active);
|
|
}
|
|
}
|
|
}
|
|
if (type2Timings && !m_nativeResolution) {
|
|
for (auto timing = type2Timings; *timing != nullptr; timing++) {
|
|
if ((*timing)->preferred && (!m_nativeResolution || m_nativeResolution->width() < (*timing)->horiz_active || m_nativeResolution->height() < (*timing)->vert_active)) {
|
|
m_nativeResolution = QSize((*timing)->horiz_active, (*timing)->vert_active);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// EDID often contains misleading information for backwards compatibility
|
|
// so only use it if we don't have the same info from DisplayID
|
|
if (const auto misc = di_edid_get_misc_features(edid); misc && !m_nativeResolution) {
|
|
if (misc->preferred_timing_is_native) {
|
|
const auto timing = di_edid_get_detailed_timing_defs(edid);
|
|
if (*timing != nullptr) {
|
|
m_nativeResolution = QSize((*timing)->horiz_video, (*timing)->vert_video);
|
|
}
|
|
}
|
|
}
|
|
|
|
m_isValid = true;
|
|
di_info_destroy(info);
|
|
}
|
|
|
|
std::optional<QSize> Edid::likelyNativeResolution() const
|
|
{
|
|
return m_nativeResolution;
|
|
}
|
|
|
|
bool Edid::isValid() const
|
|
{
|
|
return m_isValid;
|
|
}
|
|
|
|
QSize Edid::physicalSize() const
|
|
{
|
|
return m_physicalSize;
|
|
}
|
|
|
|
QByteArray Edid::eisaId() const
|
|
{
|
|
return m_eisaId;
|
|
}
|
|
|
|
QByteArray Edid::monitorName() const
|
|
{
|
|
return m_monitorName;
|
|
}
|
|
|
|
QByteArray Edid::serialNumber() const
|
|
{
|
|
return m_serialNumber;
|
|
}
|
|
|
|
QByteArray Edid::vendor() const
|
|
{
|
|
return m_vendor;
|
|
}
|
|
|
|
QByteArray Edid::raw() const
|
|
{
|
|
return m_raw;
|
|
}
|
|
|
|
QString Edid::manufacturerString() const
|
|
{
|
|
QString manufacturer;
|
|
if (!m_vendor.isEmpty()) {
|
|
manufacturer = QString::fromLatin1(m_vendor);
|
|
} else if (!m_eisaId.isEmpty()) {
|
|
manufacturer = QString::fromLatin1(m_eisaId);
|
|
}
|
|
return manufacturer;
|
|
}
|
|
|
|
QString Edid::nameString() const
|
|
{
|
|
if (!m_monitorName.isEmpty()) {
|
|
return QString::fromLatin1(m_monitorName);
|
|
} else if (!m_serialNumber.isEmpty()) {
|
|
return QString::fromLatin1(m_serialNumber);
|
|
} else {
|
|
return i18n("unknown");
|
|
}
|
|
}
|
|
|
|
QString Edid::hash() const
|
|
{
|
|
return m_hash;
|
|
}
|
|
|
|
std::optional<Colorimetry> Edid::colorimetry() const
|
|
{
|
|
return m_colorimetry;
|
|
}
|
|
|
|
double Edid::desiredMinLuminance() const
|
|
{
|
|
return m_hdrMetadata ? m_hdrMetadata->desiredContentMinLuminance : 0;
|
|
}
|
|
|
|
std::optional<double> Edid::desiredMaxFrameAverageLuminance() const
|
|
{
|
|
return m_hdrMetadata ? m_hdrMetadata->desiredMaxFrameAverageLuminance : std::nullopt;
|
|
}
|
|
|
|
std::optional<double> Edid::desiredMaxLuminance() const
|
|
{
|
|
return m_hdrMetadata ? m_hdrMetadata->desiredContentMaxLuminance : std::nullopt;
|
|
}
|
|
|
|
bool Edid::supportsPQ() const
|
|
{
|
|
return m_hdrMetadata && m_hdrMetadata->supportsPQ;
|
|
}
|
|
|
|
bool Edid::supportsBT2020() const
|
|
{
|
|
return m_hdrMetadata && m_hdrMetadata->supportsBT2020;
|
|
}
|
|
|
|
QByteArray Edid::identifier() const
|
|
{
|
|
return m_identifier;
|
|
}
|
|
|
|
} // namespace KWin
|