214 lines
6.4 KiB
C++
214 lines
6.4 KiB
C++
/**
|
|
* Copyright - See the COPYRIGHT that is included with this distribution.
|
|
* pvAccessCPP is distributed subject to a Software License Agreement found
|
|
* in file LICENSE that is included with this distribution.
|
|
*/
|
|
|
|
#include <map>
|
|
#include <vector>
|
|
|
|
#include <osiSock.h>
|
|
#include <epicsThread.h>
|
|
|
|
#include <pv/lock.h>
|
|
#include <pv/noDefaultMethods.h>
|
|
#include <pv/pvData.h>
|
|
#include <pv/reftrack.h>
|
|
#include <pv/timer.h>
|
|
|
|
#define epicsExportSharedSymbols
|
|
#include <pv/pvAccess.h>
|
|
#include <pv/factory.h>
|
|
#include "pv/codec.h"
|
|
#include <pv/serverContextImpl.h>
|
|
#include <pv/serverChannelImpl.h>
|
|
|
|
using namespace epics::pvData;
|
|
using std::string;
|
|
|
|
namespace epics {
|
|
namespace pvAccess {
|
|
|
|
ChannelProviderRegistry::shared_pointer ChannelProviderRegistry::build() {
|
|
ChannelProviderRegistry::shared_pointer ret(new ChannelProviderRegistry);
|
|
return ret;
|
|
}
|
|
|
|
ChannelProvider::shared_pointer ChannelProviderRegistry::getProvider(std::string const & providerName) {
|
|
ChannelProviderFactory::shared_pointer fact(getFactory(providerName));
|
|
if(fact)
|
|
return fact->sharedInstance();
|
|
else
|
|
return ChannelProvider::shared_pointer();
|
|
}
|
|
|
|
ChannelProvider::shared_pointer ChannelProviderRegistry::createProvider(std::string const & providerName,
|
|
const std::tr1::shared_ptr<Configuration>& conf) {
|
|
ChannelProviderFactory::shared_pointer fact(getFactory(providerName));
|
|
if(fact)
|
|
return fact->newInstance(conf);
|
|
else
|
|
return ChannelProvider::shared_pointer();
|
|
}
|
|
|
|
ChannelProviderFactory::shared_pointer ChannelProviderRegistry::getFactory(std::string const & providerName)
|
|
{
|
|
Lock G(mutex);
|
|
providers_t::const_iterator iter = providers.find(providerName);
|
|
if (iter == providers.end())
|
|
return ChannelProviderFactory::shared_pointer();
|
|
else
|
|
return iter->second;
|
|
}
|
|
|
|
void ChannelProviderRegistry::getProviderNames(std::set<std::string>& names)
|
|
{
|
|
Lock G(mutex);
|
|
for (providers_t::const_iterator iter = providers.begin();
|
|
iter != providers.end(); iter++)
|
|
names.insert(iter->first);
|
|
}
|
|
|
|
|
|
bool ChannelProviderRegistry::add(const ChannelProviderFactory::shared_pointer& fact, bool replace)
|
|
{
|
|
assert(fact);
|
|
Lock G(mutex);
|
|
std::string name(fact->getFactoryName());
|
|
if(!replace && providers.find(name)!=providers.end())
|
|
return false;
|
|
providers[name] = fact;
|
|
return true;
|
|
}
|
|
|
|
namespace {
|
|
struct FunctionFactory : public ChannelProviderFactory {
|
|
const std::string pname;
|
|
epics::pvData::Mutex sharedM;
|
|
ChannelProvider::weak_pointer shared;
|
|
const ChannelProviderRegistry::factoryfn_t fn;
|
|
|
|
FunctionFactory(const std::string& name, ChannelProviderRegistry::factoryfn_t fn)
|
|
:pname(name), fn(fn)
|
|
{}
|
|
virtual ~FunctionFactory() {}
|
|
virtual std::string getFactoryName() { return pname; }
|
|
|
|
virtual ChannelProvider::shared_pointer sharedInstance()
|
|
{
|
|
epics::pvData::Lock L(sharedM);
|
|
ChannelProvider::shared_pointer ret(shared.lock());
|
|
if(!ret) {
|
|
ret = fn(std::tr1::shared_ptr<Configuration>());
|
|
shared = ret;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
virtual ChannelProvider::shared_pointer newInstance(const std::tr1::shared_ptr<Configuration>& conf)
|
|
{
|
|
return fn(conf);
|
|
}
|
|
|
|
};
|
|
}//namespace
|
|
|
|
ChannelProviderFactory::shared_pointer ChannelProviderRegistry::add(const std::string& name, factoryfn_t fn, bool replace)
|
|
{
|
|
ChannelProviderFactory::shared_pointer F(new FunctionFactory(name, fn));
|
|
return add(F, replace) ? F : ChannelProviderFactory::shared_pointer();
|
|
}
|
|
|
|
ChannelProviderFactory::shared_pointer ChannelProviderRegistry::remove(const std::string& name)
|
|
{
|
|
Lock G(mutex);
|
|
ChannelProviderFactory::shared_pointer fact(getFactory(name));
|
|
if(fact) {
|
|
remove(fact);
|
|
}
|
|
return fact;
|
|
}
|
|
|
|
bool ChannelProviderRegistry::remove(const ChannelProviderFactory::shared_pointer& fact)
|
|
{
|
|
assert(fact);
|
|
Lock G(mutex);
|
|
providers_t::iterator iter(providers.find(fact->getFactoryName()));
|
|
if(iter!=providers.end() && iter->second==fact) {
|
|
providers.erase(iter);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
void ChannelProviderRegistry::clear()
|
|
{
|
|
Lock G(mutex);
|
|
providers.clear();
|
|
}
|
|
|
|
namespace {
|
|
struct providerRegGbl_t {
|
|
ChannelProviderRegistry::shared_pointer clients,
|
|
servers;
|
|
providerRegGbl_t()
|
|
:clients(ChannelProviderRegistry::build())
|
|
,servers(ChannelProviderRegistry::build())
|
|
{}
|
|
} *providerRegGbl;
|
|
|
|
epicsThreadOnceId providerRegOnce = EPICS_THREAD_ONCE_INIT;
|
|
|
|
void providerRegInit(void*)
|
|
{
|
|
providerRegGbl = new providerRegGbl_t;
|
|
registerRefCounter("ServerContextImpl", &ServerContextImpl::num_instances);
|
|
registerRefCounter("ServerChannel", &ServerChannel::num_instances);
|
|
registerRefCounter("BlockingTCPTransportCodec", &detail::BlockingTCPTransportCodec::num_instances);
|
|
registerRefCounter("ChannelProvider (ABC)", &ChannelProvider::num_instances);
|
|
registerRefCounter("Channel (ABC)", &Channel::num_instances);
|
|
registerRefCounter("ChannelRequester (ABC)", &ChannelRequester::num_instances);
|
|
registerRefCounter("ChannelBaseRequester (ABC)", &ChannelBaseRequester::num_instances);
|
|
registerRefCounter("ChannelRequest (ABC)", &ChannelRequest::num_instances);
|
|
registerRefCounter("ResponseHandler (ABC)", &ResponseHandler::num_instances);
|
|
}
|
|
|
|
} // namespace
|
|
|
|
ChannelProviderRegistry::shared_pointer ChannelProviderRegistry::clients()
|
|
{
|
|
epicsThreadOnce(&providerRegOnce, &providerRegInit, 0);
|
|
|
|
return providerRegGbl->clients;
|
|
}
|
|
|
|
ChannelProviderRegistry::shared_pointer ChannelProviderRegistry::servers()
|
|
{
|
|
epicsThreadOnce(&providerRegOnce, &providerRegInit, 0);
|
|
|
|
return providerRegGbl->servers;
|
|
}
|
|
|
|
ChannelFind::shared_pointer
|
|
ChannelProvider::channelList(ChannelListRequester::shared_pointer const & requester)
|
|
{
|
|
ChannelFind::shared_pointer ret;
|
|
requester->channelListResult(Status::error("not implemented"),
|
|
ret,
|
|
epics::pvData::PVStringArray::const_svector(),
|
|
false);
|
|
return ret;
|
|
}
|
|
|
|
Channel::shared_pointer
|
|
ChannelProvider::createChannel(std::string const & name,
|
|
ChannelRequester::shared_pointer const & requester,
|
|
short priority)
|
|
{
|
|
return createChannel(name, requester, priority, "");
|
|
}
|
|
|
|
}
|
|
}
|
|
|