В настоящее время я пытаюсь внедрить FFmpeg в SFML, поэтому у меня есть более широкий диапазон аудиофайлов для чтения, но я получаю ошибку [mov,mp4,m4a,3gp,3g2,mj2 @ #] moov atom not found
при открытии файла m4a. Теперь это происходит только тогда, когда я использую собственный IOContext для чтения файла вместо открытия его по URL. Эта страница здесь говорится, что я не должен использовать потоки для открытия файлов m4a, но считается ли IOContext потоком? Потому что я не могу открыть его как URL, так как работает SFML.
// Explanation of InputStream class
class InputStream {
int64_t getSize()
int64_t read(void* data, int64_t size);
int64_t seek(int64_t position);
int64_t tell(); // Gets the stream position
};
// Used for IOContext
int read(void* opaque, uint8_t* buf, int buf_size) {
sf::InputStream* stream = (sf::InputStream*)opaque;
return (int)stream->read(buf, buf_size);
}
// Used for IOContext
int64_t seek(void* opaque, int64_t offset, int whence) {
sf::InputStream* stream = (sf::InputStream*)opaque;
switch (whence) {
case SEEK_SET:
break;
case SEEK_CUR:
offset += stream->tell();
break;
case SEEK_END:
offset = stream->getSize() - offset;
}
return (int64_t)stream->seek(offset);
}
bool open(sf::InputStream& stream) {
AVFormatContext* m_formatContext = NULL;
AVIOContext* m_ioContext = NULL;
uint8_t* m_ioContextBuffer = NULL;
size_t m_ioContextBufferSize = 0;
av_register_all();
avformat_network_init();
m_formatContext = avformat_alloc_context();
m_ioContextBuffer = (uint8_t*)av_malloc(m_ioContextBufferSize);
if (!m_ioContextBuffer) {
close();
return false;
}
m_ioContext = avio_alloc_context(
m_ioContextBuffer, m_ioContextBufferSize,
0, &stream, &::read, NULL, &::seek
);
if (!m_ioContext) {
close();
return false;
}
m_formatContext = avformat_alloc_context();
m_formatContext->pb = m_ioContext;
if (avformat_open_input(&m_formatContext, NULL, NULL, NULL) != 0) {
// FAILS HERE
close();
return false;
}
//...
return true;
}
Оказывается, была только одна проблема, и это было с моей функцией поиска. Очевидно, у ffmpeg есть еще одна опция откуда AVSEEK_SIZE
, Вот реализация. После этого это работает.
int64_t seek(void* opaque, int64_t offset, int whence) {
sf::InputStream* stream = (sf::InputStream*)opaque;
switch (whence) {
case SEEK_SET:
break;
case SEEK_CUR:
offset += stream->tell();
break;
case SEEK_END:
offset = stream->getSize() - offset;
break;
case AVSEEK_SIZE:
return (int64_t)stream->getSize();
}
return (int64_t)stream->seek(offset);
}
Других решений пока нет …