Add QtAV Output

This commit is contained in:
Florian Märkl 2019-06-08 15:34:44 +02:00
commit 26b2d57fcb
No known key found for this signature in database
GPG key ID: 125BC8A5A6A1E857
6 changed files with 190 additions and 6 deletions

56
cmake/FindQtAV.cmake Normal file
View file

@ -0,0 +1,56 @@
# - Try to find the QtAV library
#
# Once done this will define
#
# QTAV_FOUND - system has libqtav
# QTAV_INCLUDE_DIRS - the libqtav include directory
# QTAV_LIBRARIES - Link these to use libqtav
find_package(Qt5 QUIET REQUIRED NO_MODULE COMPONENTS Core)
get_target_property(qmake Qt5::qmake LOCATION)
execute_process(
COMMAND ${qmake} -query QT_INSTALL_HEADERS
OUTPUT_VARIABLE QT_INSTALL_HEADERS
OUTPUT_STRIP_TRAILING_WHITESPACE
)
execute_process(
COMMAND ${qmake} -query QT_INSTALL_LIBS
OUTPUT_VARIABLE QT_INSTALL_LIBS
OUTPUT_STRIP_TRAILING_WHITESPACE
)
find_path(QTAV_INCLUDE_DIR NAMES QtAV.h
HINTS ${QT_INSTALL_HEADERS}
PATH_SUFFIXES QtAV
)
find_library(QTAV_LIBRARY NAMES QtAV QtAV1
HINTS ${QT_INSTALL_LIBS}
)
find_path(QTAVWIDGETS_INCLUDE_DIR NAMES QtAVWidgets.h
HINTS ${QT_INSTALL_HEADERS}
PATH_SUFFIXES QtAVWidgets
)
find_library(QTAVWIDGETS_LIBRARY NAMES QtAVWidgets QtAVWidgets1
HINTS ${QT_INSTALL_LIBS}
)
set(QTAV_INCLUDE_DIRS ${QTAV_INCLUDE_DIR} ${QTAV_INCLUDE_DIR}/..)
set(QTAV_LIBRARIES ${QTAV_LIBRARY})
if(NOT QTAVWIDGETS_INCLUDE_DIR MATCHES "QTAVWIDGETS_INCLUDE_DIR-NOTFOUND")
set(QTAVWIDGETS_INCLUDE_DIRS ${QTAVWIDGETS_INCLUDE_DIR} ${QTAVWIDGETS_INCLUDE_DIR}/.. ${QTAV_INCLUDE_DIRS})
endif()
if(NOT QTAV_LIBRARIES MATCHES "QTAV_LIBRARIES-NOTFOUND")
set(QTAVWIDGETS_LIBRARIES ${QTAVWIDGETS_LIBRARY} ${QTAV_LIBRARY})
endif()
find_package(PackageHandleStandardArgs REQUIRED)
find_package_handle_standard_args(QtAV REQUIRED_VARS QTAV_LIBRARIES QTAV_INCLUDE_DIRS)
mark_as_advanced(QTAV_INCLUDE_DIRS QTAV_LIBRARIES QTAVWIDGETS_INCLUDE_DIRS QTAVWIDGETS_LIBRARIES)
message("QtAV_FOUND = ${QTAV_FOUND}")
message("QTAV_INCLUDE_DIRS = ${QTAV_INCLUDE_DIRS}")
message("QTAV_LIBRARIES = ${QTAV_LIBRARIES}")
message("QTAVWIDGETS_INCLUDE_DIRS = ${QTAVWIDGETS_INCLUDE_DIRS}")
message("QTAVWIDGETS_LIBRARIES = ${QTAVWIDGETS_LIBRARIES}")

View file

@ -3,7 +3,9 @@ set(CMAKE_INCLUDE_CURRENT_DIR ON)
set(CMAKE_AUTOMOC ON) set(CMAKE_AUTOMOC ON)
set(CMAKE_AUTOUIC ON) set(CMAKE_AUTOUIC ON)
set(CMAKE_AUTORCC ON) set(CMAKE_AUTORCC ON)
find_package(Qt5 REQUIRED COMPONENTS Core Widgets Gui Multimedia) find_package(Qt5 REQUIRED COMPONENTS Core Widgets Gui Multimedia OpenGL)
find_package(QtAV REQUIRED)
add_executable(chiaki add_executable(chiaki
src/main.cpp src/main.cpp
@ -13,4 +15,7 @@ target_include_directories(chiaki PRIVATE include)
target_link_libraries(chiaki chiaki-lib) target_link_libraries(chiaki chiaki-lib)
target_link_libraries(chiaki Qt5::Core Qt5::Widgets Qt5::Gui Qt5::Multimedia) target_link_libraries(chiaki Qt5::Core Qt5::Widgets Qt5::Gui Qt5::OpenGL Qt5::Multimedia)
target_include_directories(chiaki PRIVATE ${QTAV_INCLUDE_DIRS} ${QTAVWIDGETS_INCLUDE_DIRS})
target_link_libraries(chiaki ${QTAV_LIBRARIES} ${QTAVWIDGETS_LIBRARIES})

View file

@ -20,6 +20,80 @@
#include <QMainWindow> #include <QMainWindow>
namespace QtAV
{
class VideoOutput;
class AVPlayer;
}
#include <QBuffer>
#include <QMutex>
#include <QThread>
#include <QFile>
class StreamRelayIODevice : public QIODevice
{
private:
QMutex *mutex;
QByteArray buffer;
public:
explicit StreamRelayIODevice(QObject *parent = nullptr)
: QIODevice(parent),
mutex(new QMutex(QMutex::Recursive))
{
setOpenMode(OpenModeFlag::ReadOnly);
}
~StreamRelayIODevice() override = default;
void PushSample(uint8_t *buf, size_t size)
{
{
QMutexLocker locker(mutex);
printf("push sample %zu\n", size);
buffer.append(reinterpret_cast<const char *>(buf), static_cast<int>(size));
}
emit readyRead();
}
qint64 pos() const override { return 0; }
bool open(QIODevice::OpenMode mode) override { return true; }
bool isSequential() const override { return true; }
qint64 bytesAvailable() const override
{
QMutexLocker locker(mutex);
return buffer.size() + QIODevice::bytesAvailable();
}
protected:
qint64 readData(char *data, qint64 maxSize)
{
while(true)
{
{
QMutexLocker locker(mutex);
if(buffer.size() >= maxSize)
{
printf("read %lld\n", maxSize);
memcpy(data, buffer.constData(), maxSize);
buffer.remove(0, maxSize);
return maxSize;
}
}
QThread::msleep(100);
}
}
qint64 writeData(const char *data, qint64 maxSize)
{
return -1;
}
};
class StreamWindow: public QMainWindow class StreamWindow: public QMainWindow
{ {
Q_OBJECT Q_OBJECT
@ -27,6 +101,15 @@ class StreamWindow: public QMainWindow
public: public:
explicit StreamWindow(QWidget *parent = nullptr); explicit StreamWindow(QWidget *parent = nullptr);
~StreamWindow(); ~StreamWindow();
StreamRelayIODevice *GetIODevice() { return io_device; }
private:
QtAV::VideoOutput *video_output;
QtAV::AVPlayer *av_player;
StreamRelayIODevice *io_device;
}; };
#endif // CHIAKI_GUI_STREAMWINDOW_H #endif // CHIAKI_GUI_STREAMWINDOW_H

View file

@ -21,6 +21,13 @@ void audio_frame_cb(int16_t *buf, size_t samples_count, void *user)
audio_io->write((const char *)buf, static_cast<qint64>(samples_count * 2 * 2)); audio_io->write((const char *)buf, static_cast<qint64>(samples_count * 2 * 2));
} }
void video_sample_cb(uint8_t *buf, size_t buf_size, void *user)
{
//StreamRelayIODevice *io_device = reinterpret_cast<StreamRelayIODevice *>(user);
//io_device->PushSample(buf, buf_size);
}
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
if(argc != 7) if(argc != 7)
@ -64,8 +71,6 @@ int main(int argc, char *argv[])
window.resize(640, 360); window.resize(640, 360);
window.show(); window.show();
return app.exec();
QAudioFormat audio_format; QAudioFormat audio_format;
audio_format.setSampleRate(48000); audio_format.setSampleRate(48000);
@ -87,11 +92,15 @@ int main(int argc, char *argv[])
ChiakiSession session; ChiakiSession session;
chiaki_session_init(&session, &connect_info); chiaki_session_init(&session, &connect_info);
chiaki_session_set_audio_frame_cb(&session, audio_frame_cb, NULL); chiaki_session_set_audio_frame_cb(&session, audio_frame_cb, NULL);
chiaki_session_set_video_sample_cb(&session, video_sample_cb, window.GetIODevice());
chiaki_session_start(&session); chiaki_session_start(&session);
int ret = app.exec();
chiaki_session_join(&session); chiaki_session_join(&session);
chiaki_session_fini(&session); chiaki_session_fini(&session);
delete audio_out; delete audio_out;
return 0; return ret;
} }

View file

@ -17,9 +17,40 @@
#include <streamwindow.h> #include <streamwindow.h>
#include <QtAV>
#include <QtAVWidgets>
#include <QProcessEnvironment>
StreamWindow::StreamWindow(QWidget *parent) StreamWindow::StreamWindow(QWidget *parent)
: QMainWindow(parent) : QMainWindow(parent)
{ {
video_output = new QtAV::VideoOutput(QtAV::VideoRendererId_GLWidget2, this);
setCentralWidget(video_output->widget());
av_player = new QtAV::AVPlayer(this);
connect(av_player, &QtAV::AVPlayer::stateChanged, this, [](QtAV::AVPlayer::State state){
printf("state changed to %d\n", state);
});
av_player->setRenderer(video_output);
io_device = new StreamRelayIODevice(this);
io_device->open(QIODevice::ReadOnly);
QString test_filename = QProcessEnvironment::systemEnvironment().value("CHIAKI_TEST_VIDEO");
if(!test_filename.isEmpty())
{
QFile *test_file = new QFile(test_filename, this);
test_file->open(QIODevice::OpenModeFlag::ReadOnly);
QByteArray sample = test_file->readAll();
io_device->PushSample((uint8_t *)sample.constData(), (size_t)sample.size());
test_file->close();
av_player->setIODevice(io_device);
av_player->play();
}
} }
StreamWindow::~StreamWindow() StreamWindow::~StreamWindow()

View file

@ -570,7 +570,7 @@ static void nagare_takion_av(ChiakiTakionAVPacket *header, uint8_t *buf, size_t
if(header->codec == 5/*buf[0] == 0xf4 && buf_size >= 0x50*/) if(header->codec == 5/*buf[0] == 0xf4 && buf_size >= 0x50*/)
{ {
//CHIAKI_LOGD(nagare->log, "audio!\n"); //CHIAKI_LOGD(nagare->log, "audio!\n");
chiaki_audio_receiver_frame_packet(nagare->session->audio_receiver, buf, 0x50); // TODO: why 0x50? chiaki_audio_receiver_frame_packet(nagare->session->audio_receiver, buf, 0x50); // TODO: why 0x50? this is dangerous!!!
} }
else else
{ {