You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
264 lines
10 KiB
264 lines
10 KiB
/* |
|
* Copyright © 2018-2020 Red Hat, Inc |
|
* Copyright © 2020 Aleix Pol Gonzalez <aleixpol@kde.org> |
|
* |
|
* This program is free software; you can redistribute it and/or |
|
* modify it under the terms of the GNU Lesser General Public |
|
* License as published by the Free Software Foundation; either |
|
* version 2 of the License, or (at your option) any later version. |
|
* |
|
* This library is distributed in the hope that it will be useful, |
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU |
|
* Lesser General Public License for more details. |
|
* |
|
* You should have received a copy of the GNU Lesser General Public |
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>. |
|
* |
|
* Authors: |
|
* Jan Grulich <jgrulich@redhat.com> |
|
* Aleix Pol Gonzalez <aleixpol@kde.org> |
|
*/ |
|
|
|
#include "pipewiresourcestream.h" |
|
#include "logging.h" |
|
#include "pipewirecore.h" |
|
|
|
#include <fcntl.h> |
|
#include <libdrm/drm_fourcc.h> |
|
#include <spa/utils/result.h> |
|
#include <sys/ioctl.h> |
|
#include <sys/mman.h> |
|
#include <unistd.h> |
|
|
|
#include <QLoggingCategory> |
|
#include <QOpenGLTexture> |
|
#include <QSocketNotifier> |
|
|
|
#include <KLocalizedString> |
|
|
|
void PipeWireSourceStream::onStreamStateChanged(void *data, pw_stream_state old, pw_stream_state state, const char *error_message) |
|
{ |
|
PipeWireSourceStream *pw = static_cast<PipeWireSourceStream *>(data); |
|
qCDebug(PIPEWIRE_LOGGING) << "state changed" << pw_stream_state_as_string(old) << "->" << pw_stream_state_as_string(state) << error_message; |
|
|
|
switch (state) { |
|
case PW_STREAM_STATE_ERROR: |
|
qCWarning(PIPEWIRE_LOGGING) << "Stream error: " << error_message; |
|
break; |
|
case PW_STREAM_STATE_PAUSED: |
|
Q_EMIT pw->streamReady(); |
|
break; |
|
case PW_STREAM_STATE_STREAMING: |
|
Q_EMIT pw->startStreaming(); |
|
break; |
|
case PW_STREAM_STATE_CONNECTING: |
|
break; |
|
case PW_STREAM_STATE_UNCONNECTED: |
|
if (!pw->m_stopped) { |
|
Q_EMIT pw->stopStreaming(); |
|
} |
|
break; |
|
} |
|
} |
|
|
|
void PipeWireSourceStream::onStreamParamChanged(void *data, uint32_t id, const struct spa_pod *format) |
|
{ |
|
if (!format || id != SPA_PARAM_Format) { |
|
return; |
|
} |
|
|
|
PipeWireSourceStream *pw = static_cast<PipeWireSourceStream *>(data); |
|
spa_format_video_raw_parse(format, &pw->videoFormat); |
|
|
|
const int32_t width = pw->videoFormat.size.width; |
|
const int32_t height = pw->videoFormat.size.height; |
|
const int bpp = pw->videoFormat.format == SPA_VIDEO_FORMAT_RGB || pw->videoFormat.format == SPA_VIDEO_FORMAT_BGR ? 3 : 4; |
|
const quint32 stride = SPA_ROUND_UP_N(width * bpp, 4); |
|
qCDebug(PIPEWIRE_LOGGING) << "Stream format changed"; |
|
const int32_t size = height * stride; |
|
|
|
uint8_t paramsBuffer[1024]; |
|
spa_pod_builder pod_builder = SPA_POD_BUILDER_INIT(paramsBuffer, sizeof(paramsBuffer)); |
|
|
|
const spa_pod *param = (spa_pod *)spa_pod_builder_add_object(&pod_builder, |
|
SPA_TYPE_OBJECT_ParamBuffers, |
|
SPA_PARAM_Buffers, |
|
SPA_PARAM_BUFFERS_buffers, |
|
SPA_POD_CHOICE_RANGE_Int(16, 2, 16), |
|
SPA_PARAM_BUFFERS_blocks, |
|
SPA_POD_Int(1), |
|
SPA_PARAM_BUFFERS_size, |
|
SPA_POD_Int(size), |
|
SPA_PARAM_BUFFERS_stride, |
|
SPA_POD_CHOICE_RANGE_Int(stride, stride, INT32_MAX), |
|
SPA_PARAM_BUFFERS_align, |
|
SPA_POD_Int(16)); |
|
pw_stream_update_params(pw->pwStream, ¶m, 1); |
|
} |
|
|
|
static void onProcess(void *data) |
|
{ |
|
PipeWireSourceStream *stream = static_cast<PipeWireSourceStream *>(data); |
|
stream->process(); |
|
} |
|
|
|
PipeWireSourceStream::PipeWireSourceStream(QObject *parent) |
|
: QObject(parent) |
|
{ |
|
pwStreamEvents.version = PW_VERSION_STREAM_EVENTS; |
|
pwStreamEvents.process = &onProcess; |
|
pwStreamEvents.state_changed = &PipeWireSourceStream::onStreamStateChanged; |
|
pwStreamEvents.param_changed = &PipeWireSourceStream::onStreamParamChanged; |
|
} |
|
|
|
PipeWireSourceStream::~PipeWireSourceStream() |
|
{ |
|
m_stopped = true; |
|
if (pwStream) { |
|
pw_stream_destroy(pwStream); |
|
} |
|
} |
|
|
|
uint PipeWireSourceStream::framerate() |
|
{ |
|
if (pwStream) { |
|
return videoFormat.max_framerate.num / videoFormat.max_framerate.denom; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
uint PipeWireSourceStream::nodeId() |
|
{ |
|
return pwNodeId; |
|
} |
|
|
|
bool PipeWireSourceStream::createStream(uint nodeid) |
|
{ |
|
pwCore = PipeWireCore::self(); |
|
if (!pwCore->m_error.isEmpty()) { |
|
m_error = pwCore->m_error; |
|
return false; |
|
} |
|
|
|
connect(pwCore.data(), &PipeWireCore::pipewireFailed, this, &PipeWireSourceStream::coreFailed); |
|
|
|
pwStream = pw_stream_new(pwCore->pwCore, "plasma-screencast", nullptr); |
|
pwNodeId = nodeid; |
|
pw_stream_add_listener(pwStream, &streamListener, &pwStreamEvents, this); |
|
|
|
uint8_t buffer[1024]; |
|
spa_pod_builder podBuilder = SPA_POD_BUILDER_INIT(buffer, sizeof(buffer)); |
|
spa_fraction minFramerate = SPA_FRACTION(1, 1); |
|
spa_fraction maxFramerate = SPA_FRACTION(25, 1); |
|
|
|
const spa_pod *param = (spa_pod *)spa_pod_builder_add_object(&podBuilder, |
|
SPA_TYPE_OBJECT_Format, |
|
SPA_PARAM_EnumFormat, |
|
SPA_FORMAT_mediaType, |
|
SPA_POD_Id(SPA_MEDIA_TYPE_video), |
|
SPA_FORMAT_mediaSubtype, |
|
SPA_POD_Id(SPA_MEDIA_SUBTYPE_raw), |
|
SPA_FORMAT_VIDEO_format, |
|
SPA_POD_CHOICE_ENUM_Id(6, |
|
SPA_VIDEO_FORMAT_RGBx, |
|
SPA_VIDEO_FORMAT_RGBA, |
|
SPA_VIDEO_FORMAT_BGRx, |
|
SPA_VIDEO_FORMAT_BGRA, |
|
SPA_VIDEO_FORMAT_RGB, |
|
SPA_VIDEO_FORMAT_BGR), |
|
SPA_FORMAT_VIDEO_maxFramerate, |
|
SPA_POD_CHOICE_RANGE_Fraction(&maxFramerate, &minFramerate, &maxFramerate)); |
|
|
|
pw_stream_flags s = (pw_stream_flags)(PW_STREAM_FLAG_DONT_RECONNECT | PW_STREAM_FLAG_AUTOCONNECT); |
|
if (pw_stream_connect(pwStream, PW_DIRECTION_INPUT, pwNodeId, s, ¶m, 1) != 0) { |
|
qCWarning(PIPEWIRE_LOGGING) << "Could not connect to stream"; |
|
pw_stream_destroy(pwStream); |
|
return false; |
|
} |
|
return true; |
|
} |
|
|
|
void PipeWireSourceStream::handleFrame(struct pw_buffer *buffer) |
|
{ |
|
spa_buffer *spaBuffer = buffer->buffer; |
|
|
|
if (spaBuffer->datas->chunk->size == 0) { |
|
return; |
|
} |
|
|
|
if (spaBuffer->datas->type == SPA_DATA_MemFd) { |
|
uint8_t *map = |
|
static_cast<uint8_t *>(mmap(nullptr, spaBuffer->datas->maxsize + spaBuffer->datas->mapoffset, PROT_READ, MAP_PRIVATE, spaBuffer->datas->fd, 0)); |
|
|
|
if (map == MAP_FAILED) { |
|
qCWarning(PIPEWIRE_LOGGING) << "Failed to mmap the memory: " << strerror(errno); |
|
return; |
|
} |
|
const QImage::Format format = spaBuffer->datas->chunk->stride / videoFormat.size.width == 3 ? QImage::Format_RGB888 : QImage::Format_ARGB32; |
|
|
|
QImage img(map, videoFormat.size.width, videoFormat.size.height, spaBuffer->datas->chunk->stride, format); |
|
Q_EMIT imageTextureReceived(img.copy()); |
|
|
|
munmap(map, spaBuffer->datas->maxsize + spaBuffer->datas->mapoffset); |
|
} else if (spaBuffer->datas->type == SPA_DATA_DmaBuf) { |
|
QVector<DmaBufPlane> planes; |
|
planes.reserve(spaBuffer->n_datas); |
|
for (uint i = 0; i < spaBuffer->n_datas; ++i) { |
|
const auto &data = spaBuffer->datas[i]; |
|
|
|
DmaBufPlane plane; |
|
plane.fd = data.fd; |
|
plane.stride = data.chunk->stride; |
|
plane.offset = data.chunk->offset; |
|
plane.modifier = DRM_FORMAT_MOD_INVALID; |
|
planes += plane; |
|
} |
|
Q_EMIT dmabufTextureReceived(planes, DRM_FORMAT_ARGB8888); |
|
} else if (spaBuffer->datas->type == SPA_DATA_MemPtr) { |
|
QImage img(static_cast<uint8_t *>(spaBuffer->datas->data), |
|
videoFormat.size.width, |
|
videoFormat.size.height, |
|
spaBuffer->datas->chunk->stride, |
|
QImage::Format_ARGB32); |
|
Q_EMIT imageTextureReceived(img); |
|
} else { |
|
qWarning() << "unsupported buffer type" << spaBuffer->datas->type; |
|
QImage errorImage(200, 200, QImage::Format_ARGB32_Premultiplied); |
|
errorImage.fill(Qt::red); |
|
Q_EMIT imageTextureReceived(errorImage); |
|
} |
|
} |
|
|
|
void PipeWireSourceStream::coreFailed(const QString &errorMessage) |
|
{ |
|
m_error = errorMessage; |
|
Q_EMIT stopStreaming(); |
|
} |
|
|
|
void PipeWireSourceStream::process() |
|
{ |
|
pw_buffer *buf = pw_stream_dequeue_buffer(pwStream); |
|
if (!buf) { |
|
return; |
|
} |
|
|
|
handleFrame(buf); |
|
|
|
pw_stream_queue_buffer(pwStream, buf); |
|
} |
|
|
|
void PipeWireSourceStream::stop() |
|
{ |
|
if (!m_stopped) |
|
pw_stream_set_active(pwStream, false); |
|
m_stopped = true; |
|
delete this; |
|
} |
|
|
|
void PipeWireSourceStream::setActive(bool active) |
|
{ |
|
Q_ASSERT(pwStream); |
|
pw_stream_set_active(pwStream, active); |
|
}
|
|
|