backends/drm: don't break crtc<->plane connections
The kernel rejects such commits without the crtc being completely turned off first. BUG: 462214
This commit is contained in:
parent
5fae6517ae
commit
ffad2dd7ad
1 changed files with 27 additions and 10 deletions
|
@ -170,27 +170,44 @@ void DrmGpu::initDrmResources()
|
||||||
QVector<DrmPlane *> assignedPlanes;
|
QVector<DrmPlane *> assignedPlanes;
|
||||||
for (int i = 0; i < resources->count_crtcs; ++i) {
|
for (int i = 0; i < resources->count_crtcs; ++i) {
|
||||||
uint32_t crtcId = resources->crtcs[i];
|
uint32_t crtcId = resources->crtcs[i];
|
||||||
DrmPlane *primary = nullptr;
|
QVector<DrmPlane *> primaryCandidates;
|
||||||
DrmPlane *cursor = nullptr;
|
QVector<DrmPlane *> cursorCandidates;
|
||||||
for (const auto &plane : m_planes) {
|
for (const auto &plane : m_planes) {
|
||||||
if (plane->isCrtcSupported(i) && !assignedPlanes.contains(plane.get())) {
|
if (plane->isCrtcSupported(i) && !assignedPlanes.contains(plane.get())) {
|
||||||
if (plane->type() == DrmPlane::TypeIndex::Primary) {
|
if (plane->type() == DrmPlane::TypeIndex::Primary) {
|
||||||
if (!primary || primary->getProp(DrmPlane::PropertyIndex::CrtcId)->pending() == crtcId) {
|
primaryCandidates.push_back(plane.get());
|
||||||
primary = plane.get();
|
|
||||||
}
|
|
||||||
} else if (plane->type() == DrmPlane::TypeIndex::Cursor) {
|
} else if (plane->type() == DrmPlane::TypeIndex::Cursor) {
|
||||||
if (!cursor || cursor->getProp(DrmPlane::PropertyIndex::CrtcId)->pending() == crtcId) {
|
cursorCandidates.push_back(plane.get());
|
||||||
cursor = plane.get();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
if (m_atomicModeSetting && primaryCandidates.empty()) {
|
||||||
if (m_atomicModeSetting && !primary) {
|
|
||||||
qCWarning(KWIN_DRM) << "Could not find a suitable primary plane for crtc" << resources->crtcs[i];
|
qCWarning(KWIN_DRM) << "Could not find a suitable primary plane for crtc" << resources->crtcs[i];
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
const auto findBestPlane = [crtcId](const QVector<DrmPlane *> &list) {
|
||||||
|
// if the plane is already used with this crtc, prefer it
|
||||||
|
const auto connected = std::find_if(list.begin(), list.end(), [crtcId](DrmPlane *plane) {
|
||||||
|
return plane->getProp(DrmPlane::PropertyIndex::CrtcId)->pending() == crtcId;
|
||||||
|
});
|
||||||
|
if (connected != list.end()) {
|
||||||
|
return *connected;
|
||||||
|
}
|
||||||
|
// don't take away planes from other crtcs. The kernel currently rejects such commits
|
||||||
|
const auto notconnected = std::find_if(list.begin(), list.end(), [](DrmPlane *plane) {
|
||||||
|
return plane->getProp(DrmPlane::PropertyIndex::CrtcId)->pending() == 0;
|
||||||
|
});
|
||||||
|
if (notconnected == list.end()) {
|
||||||
|
return *notconnected;
|
||||||
|
}
|
||||||
|
return list.empty() ? nullptr : list.front();
|
||||||
|
};
|
||||||
|
DrmPlane *primary = findBestPlane(primaryCandidates);
|
||||||
|
DrmPlane *cursor = findBestPlane(cursorCandidates);
|
||||||
assignedPlanes.push_back(primary);
|
assignedPlanes.push_back(primary);
|
||||||
|
if (cursor) {
|
||||||
assignedPlanes.push_back(cursor);
|
assignedPlanes.push_back(cursor);
|
||||||
|
}
|
||||||
auto crtc = std::make_unique<DrmCrtc>(this, crtcId, i, primary, cursor);
|
auto crtc = std::make_unique<DrmCrtc>(this, crtcId, i, primary, cursor);
|
||||||
if (!crtc->init()) {
|
if (!crtc->init()) {
|
||||||
continue;
|
continue;
|
||||||
|
|
Loading…
Reference in a new issue