haiku/src/kits/media/PluginManager.cpp
beveloper 94722b265c removed debug output
git-svn-id: file:///srv/svn/repos/haiku/trunk/current@6814 a95241bf-73f2-0310-859d-f6bbb57e9c96
2004-02-29 18:50:56 +00:00

253 lines
5.4 KiB
C++

/*
** Copyright 2004, Marcus Overhagen. All rights reserved.
** Distributed under the terms of the OpenBeOS License.
*/
#include <Path.h>
#include <image.h>
#include <string.h>
#include "PluginManager.h"
#include "DataExchange.h"
#include "debug.h"
PluginManager _plugin_manager;
status_t
_CreateReader(Reader **reader, int32 *streamCount, media_file_format *mff, BDataIO *source)
{
TRACE("_CreateReader enter\n");
BPositionIO *seekable_source = dynamic_cast<BPositionIO *>(source);
if (seekable_source == 0) {
printf("_CreateReader: non-seekable sources not supported yet\n");
return B_ERROR;
}
// get list of available readers from, the server
server_get_readers_request request;
server_get_readers_reply reply;
if (B_OK != QueryServer(SERVER_GET_READERS, &request, sizeof(request), &reply, sizeof(reply))) {
printf("_CreateReader: can't get list of readers\n");
return B_ERROR;
}
// try each reader by calling it's Sniff function...
for (int32 i = 0; i < reply.count; i++) {
entry_ref ref = reply.ref[i];
MediaPlugin *plugin = _plugin_manager.GetPlugin(ref);
if (!plugin) {
printf("_CreateReader: GetPlugin failed\n");
return B_ERROR;
}
ReaderPlugin *readerPlugin = dynamic_cast<ReaderPlugin *>(plugin);
if (!readerPlugin) {
printf("_CreateReader: dynamic_cast failed\n");
return B_ERROR;
}
*reader = readerPlugin->NewReader();
if (! *reader) {
printf("_CreateReader: NewReader failed\n");
return B_ERROR;
}
seekable_source->Seek(0, SEEK_SET);
(*reader)->Setup(seekable_source);
if (B_OK == (*reader)->Sniff(streamCount)) {
TRACE("_CreateReader: Sniff success (%ld stream(s))\n", *streamCount);
(*reader)->GetFileFormatInfo(mff);
return B_OK;
}
// _DestroyReader(*reader);
delete *reader;
_plugin_manager.PutPlugin(plugin);
}
TRACE("_CreateReader leave\n");
return B_MEDIA_NO_HANDLER;
}
status_t
_CreateDecoder(Decoder **_decoder, const media_format &format)
{
TRACE("_CreateDecoder enter\n");
// get decoder for this format from the server
server_get_decoder_for_format_request request;
server_get_decoder_for_format_reply reply;
request.format = format;
if (B_OK != QueryServer(SERVER_GET_DECODER_FOR_FORMAT, &request, sizeof(request), &reply, sizeof(reply))) {
printf("_CreateReader: can't get decoder for format\n");
return B_ERROR;
}
MediaPlugin *plugin = _plugin_manager.GetPlugin(reply.ref);
if (!plugin) {
printf("_CreateDecoder: GetPlugin failed\n");
return B_ERROR;
}
DecoderPlugin *decoderPlugin = dynamic_cast<DecoderPlugin *>(plugin);
if (!decoderPlugin) {
printf("_CreateDecoder: dynamic_cast failed\n");
return B_ERROR;
}
*_decoder = decoderPlugin->NewDecoder(0);
if (*_decoder == NULL) {
printf("_CreateDecoder: NewDecoder() failed\n");
return B_ERROR;
}
TRACE("_CreateDecoder leave\n");
return B_OK;
}
void
_DestroyReader(Reader *reader)
{
// ToDo: must call put plugin
delete reader;
}
void
_DestroyDecoder(Decoder *decoder)
{
// ToDo: must call put plugin
delete decoder;
}
// #pragma mark -
PluginManager::PluginManager()
{
CALLED();
fLocker = new BLocker;
fPluginList = new List<plugin_info>;
}
PluginManager::~PluginManager()
{
CALLED();
while (!fPluginList->IsEmpty()) {
plugin_info *info;
fPluginList->Get(fPluginList->CountItems() - 1, &info);
printf("PluginManager: Error, unloading PlugIn %s with usecount %d\n", info->name, info->usecount);
delete info->plugin;
unload_add_on(info->image);
fPluginList->Remove(fPluginList->CountItems() - 1);
}
delete fLocker;
}
MediaPlugin *
PluginManager::GetPlugin(const entry_ref &ref)
{
fLocker->Lock();
MediaPlugin *plugin;
plugin_info *pinfo;
plugin_info info;
for (fPluginList->Rewind(); fPluginList->GetNext(&pinfo); ) {
if (0 == strcmp(ref.name, pinfo->name)) {
plugin = pinfo->plugin;
pinfo->usecount++;
fLocker->Unlock();
return plugin;
}
}
if (!LoadPlugin(ref, &info.plugin, &info.image)) {
printf("PluginManager: Error, loading PlugIn %s failed\n", ref.name);
fLocker->Unlock();
return 0;
}
strcpy(info.name, ref.name);
info.usecount = 1;
fPluginList->Insert(info);
TRACE("PluginManager: PlugIn %s loaded\n", ref.name);
plugin = info.plugin;
fLocker->Unlock();
return plugin;
}
void
PluginManager::PutPlugin(MediaPlugin *plugin)
{
fLocker->Lock();
plugin_info *pinfo;
for (fPluginList->Rewind(); fPluginList->GetNext(&pinfo); ) {
if (plugin == pinfo->plugin) {
pinfo->usecount--;
if (pinfo->usecount == 0) {
delete pinfo->plugin;
unload_add_on(pinfo->image);
}
fPluginList->RemoveCurrent();
fLocker->Unlock();
return;
}
}
printf("PluginManager: Error, can't put PlugIn %p\n", plugin);
fLocker->Unlock();
}
bool
PluginManager::LoadPlugin(const entry_ref &ref, MediaPlugin **plugin, image_id *image)
{
BPath p(&ref);
TRACE("PluginManager: LoadPlugin trying to load %s\n", p.Path());
image_id id;
id = load_add_on(p.Path());
if (id < 0)
return false;
MediaPlugin *(*instantiate_plugin_func)();
if (get_image_symbol(id, "instantiate_plugin", B_SYMBOL_TYPE_TEXT, (void**)&instantiate_plugin_func) < B_OK) {
printf("PluginManager: Error, LoadPlugin can't find instantiate_plugin in %s\n", p.Path());
unload_add_on(id);
return false;
}
MediaPlugin *pl;
pl = (*instantiate_plugin_func)();
if (pl == 0) {
printf("PluginManager: Error, LoadPlugin instantiate_plugin in %s returned 0\n", p.Path());
unload_add_on(id);
return false;
}
*plugin = pl;
*image = id;
return true;
}