Refactorized avdecoder class.
Disable warnings from ffmpeg.
This commit is contained in:
@@ -17,6 +17,7 @@
|
|||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include "utils.h"
|
||||||
#include "avdecoder.h"
|
#include "avdecoder.h"
|
||||||
#include "cmaobject.h"
|
#include "cmaobject.h"
|
||||||
|
|
||||||
@@ -28,12 +29,15 @@
|
|||||||
AVDecoder::AvInit init;
|
AVDecoder::AvInit init;
|
||||||
|
|
||||||
AVDecoder::AVDecoder() :
|
AVDecoder::AVDecoder() :
|
||||||
pFormatCtx(NULL)
|
pFormatCtx(NULL), pCodecCtx(NULL), av_stream(NULL), av_codec(NULL), stream_index(-1)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
AVDecoder::~AVDecoder()
|
AVDecoder::~AVDecoder()
|
||||||
{
|
{
|
||||||
|
if(pCodecCtx) {
|
||||||
|
avcodec_close(pCodecCtx);
|
||||||
|
}
|
||||||
if(pFormatCtx) {
|
if(pFormatCtx) {
|
||||||
avformat_close_input(&pFormatCtx);
|
avformat_close_input(&pFormatCtx);
|
||||||
}
|
}
|
||||||
@@ -53,68 +57,62 @@ bool AVDecoder::open(const QString filename)
|
|||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void AVDecoder::getAudioMetadata(metadata_t &metadata)
|
bool AVDecoder::loadCodec(codec_type codec)
|
||||||
|
{
|
||||||
|
AVDictionary *opts = NULL;
|
||||||
|
|
||||||
|
if((stream_index = av_find_best_stream(pFormatCtx, (AVMediaType)codec, -1, -1, &av_codec, 0)) < 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
av_stream = pFormatCtx->streams[stream_index];
|
||||||
|
pCodecCtx = av_stream->codec;
|
||||||
|
|
||||||
|
if(avcodec_open2(pCodecCtx, av_codec, &opts) < 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
const char *AVDecoder::getMetadataEntry(const char *key, const char *default_value)
|
||||||
{
|
{
|
||||||
AVDictionaryEntry *entry;
|
AVDictionaryEntry *entry;
|
||||||
AVDictionary* file_metadata = pFormatCtx->metadata;
|
|
||||||
|
|
||||||
if((entry = av_dict_get(file_metadata, "artist", NULL, 0)) != NULL) {
|
if((entry = av_dict_get(pFormatCtx->metadata, key, NULL, 0)) != NULL) {
|
||||||
metadata.data.music.artist = strdup(entry->value);
|
return entry->value;
|
||||||
} else {
|
|
||||||
metadata.data.music.artist = strdup("");
|
|
||||||
}
|
}
|
||||||
|
return default_value;
|
||||||
|
}
|
||||||
|
|
||||||
if((entry = av_dict_get(file_metadata, "album", NULL, 0)) != NULL) {
|
void AVDecoder::getAudioMetadata(metadata_t &metadata)
|
||||||
metadata.data.music.album = strdup(entry->value);
|
{
|
||||||
|
metadata.data.music.artist = strdup(getMetadataEntry("artist", ""));
|
||||||
|
metadata.data.music.album = strdup(getMetadataEntry("album", ""));
|
||||||
|
metadata.data.music.title = strdup(getMetadataEntry("title", ""));
|
||||||
|
|
||||||
|
if(loadCodec(CODEC_AUDIO)) {
|
||||||
|
metadata.data.music.tracks->data.track_audio.bitrate = pCodecCtx->bit_rate;
|
||||||
} else {
|
} else {
|
||||||
metadata.data.music.album = strdup("");
|
metadata.data.music.tracks->data.track_audio.bitrate = pFormatCtx->bit_rate;
|
||||||
}
|
}
|
||||||
|
|
||||||
if((entry = av_dict_get(file_metadata, "title", NULL, 0)) != NULL) {
|
|
||||||
metadata.data.music.title = strdup(entry->value);
|
|
||||||
} else {
|
|
||||||
metadata.data.music.title = strdup("");
|
|
||||||
}
|
|
||||||
|
|
||||||
metadata.data.music.tracks->data.track_audio.bitrate = pFormatCtx->bit_rate;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void AVDecoder::getVideoMetadata(metadata_t &metadata)
|
void AVDecoder::getVideoMetadata(metadata_t &metadata)
|
||||||
{
|
{
|
||||||
AVDictionaryEntry *entry;
|
metadata.data.video.copyright = strdup(getMetadataEntry("copyright", ""));
|
||||||
AVDictionary* file_metadata = pFormatCtx->metadata;
|
metadata.data.video.explanation = strdup(getMetadataEntry("comments", ""));
|
||||||
|
metadata.data.video.title = strdup(getMetadataEntry("title", metadata.name));
|
||||||
|
|
||||||
if((entry = av_dict_get(file_metadata, "copyright", NULL, 0)) != NULL) {
|
if(loadCodec(CODEC_VIDEO)) {
|
||||||
metadata.data.video.copyright = strdup(entry->value);
|
|
||||||
} else {
|
|
||||||
metadata.data.video.copyright = strdup("");
|
|
||||||
}
|
|
||||||
|
|
||||||
if((entry = av_dict_get(file_metadata, "comments", NULL, 0)) != NULL) {
|
|
||||||
metadata.data.video.explanation = strdup(entry->value);
|
|
||||||
} else {
|
|
||||||
metadata.data.video.explanation = strdup("");
|
|
||||||
}
|
|
||||||
|
|
||||||
if((entry = av_dict_get(file_metadata, "title", NULL, 0)) != NULL) {
|
|
||||||
metadata.data.video.title = strdup(entry->value);
|
|
||||||
} else {
|
|
||||||
metadata.data.video.title = strdup(metadata.name);
|
|
||||||
}
|
|
||||||
|
|
||||||
metadata.data.video.tracks->data.track_video.duration = pFormatCtx->duration / 1000;
|
|
||||||
metadata.data.video.tracks->data.track_video.bitrate = pFormatCtx->bit_rate;
|
|
||||||
|
|
||||||
int stream_index;
|
|
||||||
AVCodec *codec = NULL;
|
|
||||||
|
|
||||||
if((stream_index = av_find_best_stream(pFormatCtx, AVMEDIA_TYPE_VIDEO, -1, -1, &codec, 0)) >= 0) {
|
|
||||||
AVCodecContext *pCodecCtx = pFormatCtx->streams[stream_index]->codec;
|
|
||||||
metadata.data.video.tracks->data.track_video.width = pCodecCtx->width;
|
metadata.data.video.tracks->data.track_video.width = pCodecCtx->width;
|
||||||
metadata.data.video.tracks->data.track_video.height = pCodecCtx->height;
|
metadata.data.video.tracks->data.track_video.height = pCodecCtx->height;
|
||||||
if(strcmp(codec->name, "h264") == 0) {
|
metadata.data.video.tracks->data.track_video.bitrate = pCodecCtx->bit_rate;
|
||||||
|
metadata.data.video.tracks->data.track_video.duration = pFormatCtx->duration / 1000;
|
||||||
|
|
||||||
|
if(strcmp(av_codec->name, "h264") == 0) {
|
||||||
metadata.data.video.tracks->data.track_video.codecType = CODEC_TYPE_AVC;
|
metadata.data.video.tracks->data.track_video.codecType = CODEC_TYPE_AVC;
|
||||||
} else if(strcmp(codec->name, "mpeg4") == 0) {
|
} else if(strcmp(av_codec->name, "mpeg4") == 0) {
|
||||||
metadata.data.video.tracks->data.track_video.codecType = CODEC_TYPE_MPEG4;
|
metadata.data.video.tracks->data.track_video.codecType = CODEC_TYPE_MPEG4;
|
||||||
} else {
|
} else {
|
||||||
metadata.data.video.tracks->data.track_video.codecType = 0;
|
metadata.data.video.tracks->data.track_video.codecType = 0;
|
||||||
@@ -126,10 +124,7 @@ QByteArray AVDecoder::getAudioThumbnail(int width, int height)
|
|||||||
{
|
{
|
||||||
QByteArray data;
|
QByteArray data;
|
||||||
|
|
||||||
int stream_index;
|
if(!loadCodec(CODEC_VIDEO)) {
|
||||||
AVCodec *codec = NULL;
|
|
||||||
if((stream_index = av_find_best_stream(pFormatCtx, AVMEDIA_TYPE_VIDEO, -1, -1, &codec, 0)) < 0) {
|
|
||||||
// no thumbnail
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -171,27 +166,16 @@ AVFrame *AVDecoder::getDecodedFrame(AVCodecContext *pCodecCtx, int stream_index)
|
|||||||
QByteArray AVDecoder::getVideoThumbnail(int width, int height)
|
QByteArray AVDecoder::getVideoThumbnail(int width, int height)
|
||||||
{
|
{
|
||||||
QByteArray data;
|
QByteArray data;
|
||||||
int stream_index;
|
|
||||||
AVFrame *pFrame;
|
AVFrame *pFrame;
|
||||||
AVDictionary *opts = NULL;
|
|
||||||
AVCodec *codec = NULL;
|
|
||||||
AVCodecContext *pCodecCtx = NULL;
|
|
||||||
|
|
||||||
int percentage = QSettings().value("videoThumbnailSeekPercentage", 30).toInt();
|
int percentage = QSettings().value("videoThumbnailSeekPercentage", 30).toInt();
|
||||||
|
|
||||||
if((stream_index = av_find_best_stream(pFormatCtx, AVMEDIA_TYPE_VIDEO, -1, -1, &codec, 0)) < 0) {
|
if(!loadCodec(CODEC_VIDEO)) {
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
pCodecCtx = pFormatCtx->streams[stream_index]->codec;
|
|
||||||
|
|
||||||
if(avcodec_open2(pCodecCtx, codec, &opts) < 0) {
|
|
||||||
avcodec_close(pCodecCtx);
|
|
||||||
return data;
|
return data;
|
||||||
}
|
}
|
||||||
|
|
||||||
qint64 seek_pos = pFormatCtx->duration * percentage / (AV_TIME_BASE * 100);
|
qint64 seek_pos = pFormatCtx->duration * percentage / (AV_TIME_BASE * 100);
|
||||||
qint64 frame = av_rescale(seek_pos,pFormatCtx->streams[stream_index]->time_base.den, pFormatCtx->streams[stream_index]->time_base.num);
|
qint64 frame = av_rescale(seek_pos, av_stream->time_base.den, av_stream->time_base.num);
|
||||||
|
|
||||||
if(avformat_seek_file(pFormatCtx, stream_index, 0, frame, frame, AVSEEK_FLAG_FRAME) < 0) {
|
if(avformat_seek_file(pFormatCtx, stream_index, 0, frame, frame, AVSEEK_FLAG_FRAME) < 0) {
|
||||||
avcodec_close(pCodecCtx);
|
avcodec_close(pCodecCtx);
|
||||||
|
|||||||
@@ -38,13 +38,33 @@ public:
|
|||||||
AVDecoder();
|
AVDecoder();
|
||||||
~AVDecoder();
|
~AVDecoder();
|
||||||
|
|
||||||
|
enum codec_type {CODEC_VIDEO = AVMEDIA_TYPE_VIDEO, CODEC_AUDIO = AVMEDIA_TYPE_AUDIO};
|
||||||
|
|
||||||
bool open(const QString filename);
|
bool open(const QString filename);
|
||||||
void close();
|
void close();
|
||||||
|
bool loadCodec(codec_type codec);
|
||||||
|
|
||||||
QByteArray getAudioThumbnail(int width, int height);
|
QByteArray getAudioThumbnail(int width, int height);
|
||||||
QByteArray getVideoThumbnail(int width, int height);
|
QByteArray getVideoThumbnail(int width, int height);
|
||||||
void getAudioMetadata(metadata_t &metadata);
|
void getAudioMetadata(metadata_t &metadata);
|
||||||
void getVideoMetadata(metadata_t &metadata);
|
void getVideoMetadata(metadata_t &metadata);
|
||||||
|
const char *getMetadataEntry(const char *key, const char *default_value = NULL);
|
||||||
|
|
||||||
|
inline int getWidth() {
|
||||||
|
return pCodecCtx->width;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline int getHeight() {
|
||||||
|
return pCodecCtx->height;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline int getDuration() {
|
||||||
|
return pFormatCtx->duration / 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline int getBitrate() {
|
||||||
|
return pCodecCtx->bit_rate;
|
||||||
|
}
|
||||||
|
|
||||||
// simulate a static constructor to initialize libav only once
|
// simulate a static constructor to initialize libav only once
|
||||||
class AvInit
|
class AvInit
|
||||||
@@ -52,6 +72,8 @@ public:
|
|||||||
public:
|
public:
|
||||||
AvInit() {
|
AvInit() {
|
||||||
av_register_all();
|
av_register_all();
|
||||||
|
// hide warning logs
|
||||||
|
av_log_set_level(AV_LOG_ERROR);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -61,6 +83,10 @@ private:
|
|||||||
AVFrame *getDecodedFrame(AVCodecContext *pCodecCtx, int stream_index);
|
AVFrame *getDecodedFrame(AVCodecContext *pCodecCtx, int stream_index);
|
||||||
|
|
||||||
AVFormatContext *pFormatCtx;
|
AVFormatContext *pFormatCtx;
|
||||||
|
AVCodecContext *pCodecCtx;
|
||||||
|
AVStream *av_stream;
|
||||||
|
AVCodec *av_codec;
|
||||||
|
int stream_index;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // AVDECODER_H
|
#endif // AVDECODER_H
|
||||||
|
|||||||
Reference in New Issue
Block a user