00001
00002 #include "Rivet/Rivet.hh"
00003 #include "Rivet/Tools/Logging.hh"
00004 #include "Rivet/ProjectionHandler.hh"
00005 #include "Rivet/Cmp.hh"
00006 #include <algorithm>
00007
00008 namespace Rivet {
00009
00010
00011
00012 ProjectionHandler* ProjectionHandler::_instance = 0;
00013
00014
00015
00016
00017
00018 ProjectionHandler* ProjectionHandler::create() {
00019 if (!_instance) {
00020 _instance = new ProjectionHandler();
00021 Log::getLog("Rivet.ProjectionHandler")
00022 << Log::TRACE << "Created new ProjectionHandler at " << _instance << endl;
00023 }
00024 return _instance;
00025 }
00026
00027
00028
00029
00030 Log& ProjectionHandler::getLog() const {
00031 return Log::getLog("Rivet.ProjectionHandler");
00032 }
00033
00034
00035
00036 void ProjectionHandler::clear() {
00037 _projs.clear();
00038 _namedprojs.clear();
00039 }
00040
00041
00042
00043
00044 ProjectionHandler::~ProjectionHandler() {
00045 clear();
00046 }
00047
00048
00049
00050
00051
00052
00053 const Projection& ProjectionHandler::registerProjection(const ProjectionApplier& parent,
00054 const Projection& proj,
00055 const string& name)
00056 {
00057 getLog() << Log::TRACE << "Trying to register"
00058 << " projection " << &proj << " (" << proj.name() << ")"
00059 << " for parent " << &parent << " (" << parent.name() << ")"
00060 << " with name '" << name << "'" << endl;
00061
00062
00063 const bool dupOk = _checkDuplicate(parent, proj, name);
00064 if (!dupOk) exit(1);
00065
00066
00067 const Projection* p = _getEquiv(proj);
00068 if (p == 0) {
00069
00070 p = _clone(parent, proj);
00071 }
00072
00073
00074 p = _register(parent, *p, name);
00075
00076
00077 return *p;
00078 }
00079
00080
00081
00082
00083
00084 const Projection* ProjectionHandler::registerProjection(const ProjectionApplier& parent,
00085 const Projection* proj,
00086 const string& name) {
00087 if (proj == 0) return 0;
00088 const Projection& p = registerProjection(parent, *proj, name);
00089 return &p;
00090 }
00091
00092
00093
00094
00095
00096 const Projection* ProjectionHandler::_clone(const ProjectionApplier& UNUSED(parent),
00097 const Projection& proj)
00098 {
00099
00100 getLog() << Log::TRACE << "Cloning projection " << proj.name() << " from " << &proj << endl;
00101 const Projection* newproj = proj.clone();
00102 getLog() << Log::TRACE << "Cloned projection " << proj.name() << " at " << newproj << endl;
00103
00104
00105
00106 if (&proj != newproj) {
00107 NamedProjsMap::const_iterator nps = _namedprojs.find(&proj);
00108 if (nps != _namedprojs.end()) {
00109 getLog() << Log::TRACE << "Cloning registered projections list: "
00110 << &proj << " -> " << newproj << endl;
00111 _namedprojs[newproj] = nps->second;
00112 }
00113 }
00114
00115 return newproj;
00116 }
00117
00118
00119
00120
00121
00122
00123 const Projection* ProjectionHandler::_register(const ProjectionApplier& parent,
00124 const Projection& proj,
00125 const string& name)
00126 {
00127 ProjHandle ph(&proj);
00128 getLog() << Log::TRACE << "Registering new projection at " << ph << endl;
00129
00130
00131 _projs.insert(ph);
00132
00133
00134 _namedprojs[&parent][name] = ph;
00135
00136 return ph.get();
00137 }
00138
00139
00140
00141
00142
00143 const Projection* ProjectionHandler::_getEquiv(const Projection& proj) const
00144 {
00145
00146 const std::type_info& newtype = typeid(proj);
00147 getLog() << Log::TRACE << "RTTI type of " << &proj << " is " << newtype.name() << endl;
00148
00149
00150 getLog() << Log::TRACE << "Comparing " << &proj
00151 << " with " << _projs.size()
00152 << " registered projection" << (_projs.size() == 1 ? "" : "s") << endl;
00153 foreach (const ProjHandle& ph, _projs) {
00154
00155 const std::type_info& regtype = typeid(*ph);
00156 getLog() << Log::TRACE << "RTTI type comparison with " << ph << ": "
00157 << newtype.name() << " vs. " << regtype.name() << endl;
00158 if (newtype != regtype) continue;
00159 getLog() << Log::TRACE << "RTTI type matches with " << ph << endl;
00160
00161
00162 if (pcmp(*ph, proj) != EQUIVALENT) {
00163 getLog() << Log::TRACE << "Projections at "
00164 << &proj << " and " << ph << " are not equivalent" << endl;
00165 } else {
00166 getLog() << Log::TRACE << "MATCH! Projections at "
00167 << &proj << " and " << ph << " are equivalent" << endl;
00168 return ph.get();
00169 }
00170 }
00171
00172
00173 return 0;
00174 }
00175
00176
00177
00178 string ProjectionHandler::_getStatus() const {
00179 ostringstream msg;
00180 msg << "Current projection hierarchy:" << endl;
00181 foreach (const NamedProjsMap::value_type& nps, _namedprojs) {
00182
00183 msg << nps.first << endl;
00184 foreach (const NamedProjs::value_type& np, nps.second) {
00185 msg << " " << np.second << " (" << np.second->name()
00186 << ", locally called '" << np.first << "')" << endl;
00187 }
00188 msg << endl;
00189 }
00190 return msg.str();
00191 }
00192
00193
00194
00195
00196 bool ProjectionHandler::_checkDuplicate(const ProjectionApplier& parent,
00197 const Projection& proj,
00198 const string& name) const
00199 {
00200 NamedProjsMap::const_iterator ipnps = _namedprojs.find(&parent);
00201 if (ipnps != _namedprojs.end()) {
00202 const NamedProjs pnps = ipnps->second;
00203 const NamedProjs::const_iterator ipph = pnps.find(name);
00204 if (ipph != pnps.end()) {
00205 const ProjHandle pph = ipph->second;
00206 getLog() << Log::ERROR << "Projection clash! "
00207 << parent.name() << " (" << &parent << ") "
00208 << "is trying to overwrite its registered '" << name << "' "
00209 << "projection (" << pph << "="
00210 << pph->name() << ") with a non-equivalent projection "
00211 << "(" << &proj << "=" << proj.name() << ")" << endl;
00212 getLog() << Log::ERROR << _getStatus();
00213 return false;
00214 }
00215 }
00216 return true;
00217 }
00218
00219
00220
00221
00222 void ProjectionHandler::removeProjectionApplier(ProjectionApplier& parent) {
00223 NamedProjsMap::iterator npi = _namedprojs.find(&parent);
00224 if (npi != _namedprojs.end()) _namedprojs.erase(npi);
00225
00226 const Projection* parentprojptr = dynamic_cast<Projection*>(&parent);
00227 if (parentprojptr) {
00228 ProjHandle parentph(parentprojptr);
00229 ProjHandles::iterator pi = find(_projs.begin(), _projs.end(), parentph);
00230 if (pi != _projs.end()) _projs.erase(pi);
00231 }
00232 }
00233
00234
00235
00236
00237 set<const Projection*> ProjectionHandler::getChildProjections(const ProjectionApplier& parent,
00238 ProjDepth depth) const
00239 {
00240 set<const Projection*> toplevel;
00241 NamedProjs nps = _namedprojs.find(&parent)->second;
00242 foreach (NamedProjs::value_type& np, nps) {
00243 toplevel.insert(np.second.get());
00244 }
00245 if (depth == SHALLOW) {
00246
00247 return toplevel;
00248 } else {
00249
00250 set<const Projection*> alllevels = toplevel;
00251 foreach (const Projection* p, toplevel) {
00252 set<const Projection*> allsublevels = getChildProjections(*p, DEEP);
00253 alllevels.insert(allsublevels.begin(), allsublevels.end());
00254 }
00255 return alllevels;
00256 }
00257 }
00258
00259
00260
00261
00262 const Projection& ProjectionHandler::getProjection(const ProjectionApplier& parent,
00263 const string& name) const {
00264 getLog() << Log::TRACE << "Searching for child projection '"
00265 << name << "' of " << &parent << endl;
00266 NamedProjsMap::const_iterator nps = _namedprojs.find(&parent);
00267 if (nps == _namedprojs.end()) {
00268 ostringstream msg;
00269 msg << "No projections registered for parent " << &parent;
00270 throw Error(msg.str());
00271 }
00272 NamedProjs::const_iterator np = nps->second.find(name);
00273 if (np == nps->second.end()) {
00274 ostringstream msg;
00275 msg << "No projection '" << name << "' found for parent " << &parent;
00276 throw Error(msg.str());
00277 }
00278
00279
00280 return *(np->second);
00281 }
00282
00283
00284
00285 }