[platforms/drm] At safety checks for the properties

Summary:
The AMS code accesses elements in a vector which might not be valid. This
change refactors the code to be more robust, especially the DrmPlane,
which started to crash after adding transformation support.

BUG: 386490

Reviewers: #kwin, #plasma, fvogt, subdiff

Subscribers: plasma-devel, kwin

Tags: #kwin

Differential Revision: https://phabricator.kde.org/D8752
This commit is contained in:
Martin Flöser 2017-11-10 20:38:33 +01:00
parent 320602ae14
commit c601e875cf
6 changed files with 65 additions and 32 deletions

View file

@ -41,9 +41,14 @@ DrmObject::~DrmObject()
delete p; delete p;
} }
void DrmObject::setPropertyNames(QVector<QByteArray> &&vector)
{
m_propsNames = std::move(vector);
m_props.fill(nullptr, m_propsNames.size());
}
void DrmObject::initProp(int n, drmModeObjectProperties *properties, QVector<QByteArray> enumNames) void DrmObject::initProp(int n, drmModeObjectProperties *properties, QVector<QByteArray> enumNames)
{ {
m_props.resize(m_propsNames.size());
for (unsigned int i = 0; i < properties->count_props; ++i) { for (unsigned int i = 0; i < properties->count_props; ++i) {
drmModePropertyRes *prop = drmModeGetProperty(m_backend->fd(), properties->props[i]); drmModePropertyRes *prop = drmModeGetProperty(m_backend->fd(), properties->props[i]);
if (!prop) { if (!prop) {
@ -58,10 +63,10 @@ void DrmObject::initProp(int n, drmModeObjectProperties *properties, QVector<QBy
} }
} }
bool DrmObject::atomicAddProperty(drmModeAtomicReq *req, int prop, uint64_t value) bool DrmObject::atomicAddProperty(drmModeAtomicReq *req, Property *property)
{ {
if (drmModeAtomicAddProperty(req, m_id, m_props[prop]->propId(), value) <= 0) { if (drmModeAtomicAddProperty(req, m_id, property->propId(), property->value()) <= 0) {
qCWarning(KWIN_DRM) << "Adding property" << m_propsNames[prop] << "to atomic commit failed for object" << this; qCWarning(KWIN_DRM) << "Adding property" << property->name() << "to atomic commit failed for object" << this;
return false; return false;
} }
return true; return true;
@ -72,7 +77,11 @@ bool DrmObject::atomicPopulate(drmModeAtomicReq *req)
bool ret = true; bool ret = true;
for (int i = 0; i < m_props.size(); i++) { for (int i = 0; i < m_props.size(); i++) {
ret &= atomicAddProperty(req, i, m_props[i]->value()); auto property = m_props.at(i);
if (!property) {
continue;
}
ret &= atomicAddProperty(req, property);
} }
if (!ret) { if (!ret) {

View file

@ -54,37 +54,34 @@ public:
m_output = output; m_output = output;
} }
uint32_t propId(int prop) {
return m_props[prop]->propId();
}
uint64_t value(int prop) {
return m_props[prop]->value();
}
bool propHasEnum(int prop, uint64_t value) const { bool propHasEnum(int prop, uint64_t value) const {
return m_props[prop]->hasEnum(value); auto property = m_props.at(prop);
return property ? property->hasEnum(value) : false;
} }
void setValue(int prop, uint64_t new_value) void setValue(int prop, uint64_t new_value)
{ {
Q_ASSERT(prop < m_props.size()); Q_ASSERT(prop < m_props.size());
m_props[prop]->setValue(new_value); auto property = m_props.at(prop);
if (property) {
property->setValue(new_value);
}
} }
virtual bool atomicPopulate(drmModeAtomicReq *req); virtual bool atomicPopulate(drmModeAtomicReq *req);
protected: protected:
virtual bool initProps() = 0; // only derived classes know names and quantity of properties virtual bool initProps() = 0; // only derived classes know names and quantity of properties
void setPropertyNames(QVector<QByteArray> &&vector);
void initProp(int n, drmModeObjectProperties *properties, QVector<QByteArray> enumNames = QVector<QByteArray>(0)); void initProp(int n, drmModeObjectProperties *properties, QVector<QByteArray> enumNames = QVector<QByteArray>(0));
bool atomicAddProperty(drmModeAtomicReq *req, int prop, uint64_t value); class Property;
bool atomicAddProperty(drmModeAtomicReq *req, Property *property);
DrmBackend *m_backend; DrmBackend *m_backend;
const uint32_t m_id = 0; const uint32_t m_id = 0;
DrmOutput *m_output = nullptr; DrmOutput *m_output = nullptr;
// for comparision with received name of DRM object // for comparision with received name of DRM object
QVector<QByteArray> m_propsNames;
class Property;
QVector<Property *> m_props; QVector<Property *> m_props;
class Property class Property
@ -102,15 +99,18 @@ protected:
return m_enumMap.contains(value); return m_enumMap.contains(value);
} }
uint32_t propId() { uint32_t propId() const {
return m_propId; return m_propId;
} }
uint64_t value() { uint64_t value() const {
return m_value; return m_value;
} }
void setValue(uint64_t new_value) { void setValue(uint64_t new_value) {
m_value = new_value; m_value = new_value;
} }
const QByteArray &name() const {
return m_propName;
}
private: private:
uint32_t m_propId = 0; uint32_t m_propId = 0;
@ -120,6 +120,9 @@ protected:
QVector<uint64_t> m_enumMap; QVector<uint64_t> m_enumMap;
QVector<QByteArray> m_enumNames; QVector<QByteArray> m_enumNames;
}; };
private:
QVector<QByteArray> m_propsNames;
}; };

View file

@ -51,9 +51,9 @@ bool DrmConnector::atomicInit()
bool DrmConnector::initProps() bool DrmConnector::initProps()
{ {
m_propsNames = { setPropertyNames( {
QByteArrayLiteral("CRTC_ID"), QByteArrayLiteral("CRTC_ID"),
}; });
drmModeObjectProperties *properties = drmModeObjectGetProperties(m_backend->fd(), m_id, DRM_MODE_OBJECT_CONNECTOR); drmModeObjectProperties *properties = drmModeObjectGetProperties(m_backend->fd(), m_id, DRM_MODE_OBJECT_CONNECTOR);
if (!properties) { if (!properties) {

View file

@ -48,10 +48,10 @@ bool DrmCrtc::atomicInit()
bool DrmCrtc::initProps() bool DrmCrtc::initProps()
{ {
m_propsNames = { setPropertyNames({
QByteArrayLiteral("MODE_ID"), QByteArrayLiteral("MODE_ID"),
QByteArrayLiteral("ACTIVE"), QByteArrayLiteral("ACTIVE"),
}; });
drmModeObjectProperties *properties = drmModeObjectGetProperties(m_backend->fd(), m_id, DRM_MODE_OBJECT_CRTC); drmModeObjectProperties *properties = drmModeObjectGetProperties(m_backend->fd(), m_id, DRM_MODE_OBJECT_CRTC);
if (!properties) { if (!properties) {
@ -81,6 +81,9 @@ void DrmCrtc::flipBuffer()
bool DrmCrtc::blank() bool DrmCrtc::blank()
{ {
if (!m_output) {
return false;
}
if (!m_blackBuffer) { if (!m_blackBuffer) {
DrmDumbBuffer *blackBuffer = m_backend->createBuffer(m_output->pixelSize()); DrmDumbBuffer *blackBuffer = m_backend->createBuffer(m_output->pixelSize());
if (!blackBuffer->map()) { if (!blackBuffer->map()) {

View file

@ -63,7 +63,7 @@ bool DrmPlane::atomicInit()
bool DrmPlane::initProps() bool DrmPlane::initProps()
{ {
m_propsNames = { setPropertyNames( {
QByteArrayLiteral("type"), QByteArrayLiteral("type"),
QByteArrayLiteral("SRC_X"), QByteArrayLiteral("SRC_X"),
QByteArrayLiteral("SRC_Y"), QByteArrayLiteral("SRC_Y"),
@ -76,7 +76,7 @@ bool DrmPlane::initProps()
QByteArrayLiteral("FB_ID"), QByteArrayLiteral("FB_ID"),
QByteArrayLiteral("CRTC_ID"), QByteArrayLiteral("CRTC_ID"),
QByteArrayLiteral("rotation") QByteArrayLiteral("rotation")
}; });
QVector<QByteArray> typeNames = { QVector<QByteArray> typeNames = {
QByteArrayLiteral("Primary"), QByteArrayLiteral("Primary"),
@ -129,29 +129,40 @@ bool DrmPlane::initProps()
DrmPlane::TypeIndex DrmPlane::type() DrmPlane::TypeIndex DrmPlane::type()
{ {
uint64_t v = value(int(PropertyIndex::Type)); auto property = m_props.at(int(PropertyIndex::Type));
if (!property) {
return TypeIndex::Overlay;
}
int typeCount = int(TypeIndex::Count); int typeCount = int(TypeIndex::Count);
for (int i = 0; i < typeCount; i++) { for (int i = 0; i < typeCount; i++) {
if (m_props[int(PropertyIndex::Type)]->enumMap(i) == v) { if (property->enumMap(i) == property->value()) {
return TypeIndex(i); return TypeIndex(i);
} }
} }
return TypeIndex::Overlay; return TypeIndex::Overlay;
} }
void DrmPlane::setNext(DrmBuffer *b){ void DrmPlane::setNext(DrmBuffer *b)
setValue(int(PropertyIndex::FbId), b ? b->bufferId() : 0); {
if (auto property = m_props.at(int(PropertyIndex::FbId))) {
property->setValue(b ? b->bufferId() : 0);
}
m_next = b; m_next = b;
} }
void DrmPlane::setTransformation(Transformations t) void DrmPlane::setTransformation(Transformations t)
{ {
setValue(int(PropertyIndex::Rotation), int(t)); if (auto property = m_props.at(int(PropertyIndex::Rotation))) {
property->setValue(int(t));
}
} }
DrmPlane::Transformations DrmPlane::transformation() DrmPlane::Transformations DrmPlane::transformation()
{ {
return Transformations(int(value(int(PropertyIndex::Rotation)))); if (auto property = m_props.at(int(PropertyIndex::Rotation))) {
return Transformations(int(property->value()));
}
return Transformations(Transformation::Rotate0);
} }
bool DrmPlane::atomicPopulate(drmModeAtomicReq *req) bool DrmPlane::atomicPopulate(drmModeAtomicReq *req)
@ -159,7 +170,11 @@ bool DrmPlane::atomicPopulate(drmModeAtomicReq *req)
bool ret = true; bool ret = true;
for (int i = 1; i < m_props.size(); i++) { for (int i = 1; i < m_props.size(); i++) {
ret &= atomicAddProperty(req, i, m_props[i]->value()); auto property = m_props.at(i);
if (!property) {
continue;
}
ret &= atomicAddProperty(req, property);
} }
if (!ret) { if (!ret) {

View file

@ -168,6 +168,9 @@ bool EglGbmBackend::resetOutput(Output &o, DrmOutput *drmOutput)
} else { } else {
// destroy previous surface // destroy previous surface
if (o.eglSurface != EGL_NO_SURFACE) { if (o.eglSurface != EGL_NO_SURFACE) {
if (surface() == o.eglSurface) {
setSurface(eglSurface);
}
eglDestroySurface(eglDisplay(), o.eglSurface); eglDestroySurface(eglDisplay(), o.eglSurface);
} }
o.eglSurface = eglSurface; o.eglSurface = eglSurface;