X-Git-Url: http://git.tdb.fi/?p=libs%2Fdatafile.git;a=blobdiff_plain;f=source%2Floader.cpp;h=54604dd8bebcd20abbd7892bc254aaf0b4c9ec8a;hp=8d4f2bea35b07f0cf485cbc50a8d22fe146b5cd1;hb=6fde1d2f014faa23b988d0db9d2437ff9745294b;hpb=256f7238bc60d6dcc31a564988f5cc02a60c4537 diff --git a/source/loader.cpp b/source/loader.cpp index 8d4f2be..54604dd 100644 --- a/source/loader.cpp +++ b/source/loader.cpp @@ -1,39 +1,111 @@ /* $Id$ This file is part of libmspdatafile -Copyright © 2006-2008 Mikko Rasa, Mikkosoft Productions +Copyright © 2006-2008, 2010 Mikko Rasa, Mikkosoft Productions Distributed under the LGPL */ +#include #include "loader.h" +#include "type.h" using namespace std; +namespace { + +bool signature_match(char s, char a) +{ + if(s==a) + return true; + if(s==Msp::DataFile::IntType::signature && a==Msp::DataFile::FloatType::signature) + return true; + return false; +} + +bool signature_match(const string &st_sig, const string &act_sig) +{ + if(act_sig=="*") + return true; + else if(act_sig.size()==2 && act_sig[1]=='*') + { + for(string::const_iterator i=st_sig.begin(); i!=st_sig.end(); ++i) + if(*i!=act_sig[0]) + return false; + + return true; + } + else if(st_sig.size()==act_sig.size()) + { + for(unsigned i=0; i::const_iterator i=st.sub.begin(); i!=st.sub.end(); ++i) - load_statement(*i); - finish(); + for(ActionMap::iterator i = actions.begin(); i!=actions.end(); ++i) + delete i->second; } void Loader::load(Parser &p) { while(p) { - Statement st=p.parse(); + Statement st = p.parse(); if(st.valid) load_statement(st); } finish(); } -Loader::~Loader() +void Loader::load(const Statement &st) { - for(ActionMap::iterator i=actions.begin(); i!=actions.end(); ++i) - delete i->second; + for(list::const_iterator i = st.sub.begin(); i!=st.sub.end(); ++i) + load_statement(*i); + finish(); +} + +void Loader::load_statement(const Statement &st) +{ + cur_st = &st; + + try + { + LoaderAction *act = find_action(ActionKey(st.keyword, st.get_signature())); + if(act) + { + sub_loaded = false; + act->execute(*this, st); + if(check_sub_loads && !st.sub.empty() && !sub_loaded) + throw Exception("Substatements were not loaded"); + } + } + catch(Exception &e) + { + cur_st = 0; + if(!e.where()[0]) + e.at(st.get_location()); + + throw; + } + + cur_st = 0; } void Loader::load_sub_with(Loader &ldr) @@ -42,40 +114,48 @@ void Loader::load_sub_with(Loader &ldr) throw InvalidState("load_sub called without current statement"); ldr.load(*cur_st); + sub_loaded = true; } -void Loader::add(const string &k, LoaderAction *a) +void Loader::add(const string &kwd, LoaderAction *act) { - ActionMap::iterator i=actions.find(k); + ActionKey key(kwd, (act ? act->get_signature() : "*")); + ActionMap::iterator i = actions.find(key); if(i!=actions.end()) { delete i->second; - i->second=a; + i->second = act; } else - actions[k]=a; + actions[key] = act; } -void Loader::load_statement(const Statement &st) +LoaderAction *Loader::find_action(const ActionKey &key) const { - cur_st=&st; - ActionMap::iterator j=actions.find(st.keyword); - if(j==actions.end()) - throw_at(KeyError("Unknown keyword", st.keyword), st.get_location()); - if(j->second) - { - try - { - j->second->execute(*this, st); - } - catch(Exception &e) - { - if(!e.where()[0]) - e.at(st.get_location()); - throw; - } - } - cur_st=0; + ActionMap::const_iterator begin = actions.lower_bound(ActionKey(key.keyword, string())); + ActionMap::const_iterator end = actions.upper_bound(ActionKey(key.keyword, "~")); + + if(begin==end) + throw KeyError("Unknown keyword", key.keyword); + + for(ActionMap::const_iterator i=begin; i!=end; ++i) + if(signature_match(key.signature, i->first.signature)) + return i->second; + + throw KeyError(format("Keyword '%s' does not accept signature '%s'", key.keyword, key.signature)); +} + + +Loader::ActionKey::ActionKey(const string &k, const string &s): + keyword(k), + signature(s) +{ } + +bool Loader::ActionKey::operator<(const ActionKey &other) const +{ + if(keyword!=other.keyword) + return keyword