decoder/flac: add C++ libFLAC wrappers

Not using libFLAC++ because this library adds a lot of overhead due to
virtual method calls.  This new class library is zero-overhead.
This commit is contained in:
Max Kellermann 2012-10-02 11:24:05 +02:00
parent 9c1d1ef268
commit c645b906f3
2 changed files with 71 additions and 14 deletions

View File

@ -366,27 +366,22 @@ static bool
oggflac_scan_file(const char *file, oggflac_scan_file(const char *file,
const struct tag_handler *handler, void *handler_ctx) const struct tag_handler *handler, void *handler_ctx)
{ {
FLAC__Metadata_Iterator *it; FLACMetadataChain chain;
FLAC__StreamMetadata *block; if (!chain.ReadOgg(file)) {
FLAC__Metadata_Chain *chain = FLAC__metadata_chain_new(); g_debug("Failed to read OggFLAC tags: %s",
chain.GetStatusString());
if (!(FLAC__metadata_chain_read_ogg(chain, file))) {
FLAC__metadata_chain_delete(chain);
return false; return false;
} }
it = FLAC__metadata_iterator_new(); FLACMetadataIterator iterator(chain);
FLAC__metadata_iterator_init(it, chain);
do { do {
if (!(block = FLAC__metadata_iterator_get_block(it))) FLAC__StreamMetadata *block = iterator.GetBlock();
if (block == nullptr)
break; break;
flac_scan_metadata(block, handler, handler_ctx); flac_scan_metadata(block, handler, handler_ctx);
} while (FLAC__metadata_iterator_next(it)); } while (iterator.Next());
FLAC__metadata_iterator_delete(it);
FLAC__metadata_chain_delete(chain);
return true; return true;
} }

View File

@ -20,9 +20,71 @@
#ifndef MPD_FLAC_METADATA_H #ifndef MPD_FLAC_METADATA_H
#define MPD_FLAC_METADATA_H #define MPD_FLAC_METADATA_H
#include "gcc.h"
#include <FLAC/metadata.h>
#include <assert.h> #include <assert.h>
#include <stdbool.h> #include <stdbool.h>
#include <FLAC/metadata.h>
class FLACMetadataChain {
FLAC__Metadata_Chain *chain;
public:
FLACMetadataChain():chain(::FLAC__metadata_chain_new()) {}
~FLACMetadataChain() {
::FLAC__metadata_chain_delete(chain);
}
explicit operator FLAC__Metadata_Chain *() {
return chain;
}
bool Read(const char *path) {
return ::FLAC__metadata_chain_read(chain, path);
}
bool ReadOgg(const char *path) {
return ::FLAC__metadata_chain_read_ogg(chain, path);
}
gcc_pure
FLAC__Metadata_ChainStatus GetStatus() const {
return ::FLAC__metadata_chain_status(chain);
}
gcc_pure
const char *GetStatusString() const {
return FLAC__Metadata_ChainStatusString[GetStatus()];
}
};
class FLACMetadataIterator {
FLAC__Metadata_Iterator *iterator;
public:
FLACMetadataIterator():iterator(::FLAC__metadata_iterator_new()) {}
FLACMetadataIterator(FLACMetadataChain &chain)
:iterator(::FLAC__metadata_iterator_new()) {
::FLAC__metadata_iterator_init(iterator,
(FLAC__Metadata_Chain *)chain);
}
~FLACMetadataIterator() {
::FLAC__metadata_iterator_delete(iterator);
}
bool Next() {
return ::FLAC__metadata_iterator_next(iterator);
}
gcc_pure
FLAC__StreamMetadata *GetBlock() {
return ::FLAC__metadata_iterator_get_block(iterator);
}
};
struct tag_handler; struct tag_handler;
struct tag; struct tag;