X-Git-Url: http://git.tdb.fi/?a=blobdiff_plain;f=source%2Floader.cpp;h=a05e5f0dfe187b0060cd307f21146d7f199dbb55;hb=5520420ecb8da0f6528b2681569f7504f75ddf38;hp=c2de4011232c93da747bfdead7df18af72c6087d;hpb=0d8df25704366f3576c417436c90fbac2e479632;p=libs%2Fdatafile.git diff --git a/source/loader.cpp b/source/loader.cpp index c2de401..a05e5f0 100644 --- a/source/loader.cpp +++ b/source/loader.cpp @@ -1,6 +1,6 @@ #include #include -#include "dataerror.h" +#include "except.h" #include "loader.h" #include "type.h" @@ -8,37 +8,37 @@ using namespace std; namespace { -bool signature_match(char s, char a) +int signature_match(char s, char a) { if(s==a) - return true; + return 3; if(s==Msp::DataFile::IntType::signature && a==Msp::DataFile::FloatType::signature) - return true; - return false; + return 1; + return 0; } -bool signature_match(const string &st_sig, const string &act_sig) +int signature_match(const string &st_sig, const string &act_sig) { if(act_sig=="*") - return true; + return 1; 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; + int match = 3; + for(string::const_iterator i=st_sig.begin(); (i!=st_sig.end() && match); ++i) + match = min(match, signature_match(*i, act_sig[0])); - return true; + return match; } else if(st_sig.size()==act_sig.size()) { - for(unsigned i=0; i set_parser(cur_parser, &p); + SetForScope set_level(cur_level, l); + + while(p) + { + const StatementKey *key = p.peek(l); + if(!key) + break; + + LoaderAction *act = find_action(*key); + if(act) + { + SetFlag set_direct(direct); + if(!p.parse_and_load(l, *this, *act)) + throw logic_error("direct load failed"); + } + else + load_statement(p.parse()); + } +} + void Loader::load_statement(const Statement &st) { SetForScope set_cst(cur_st, &st); @@ -134,11 +142,18 @@ void Loader::load_statement(const Statement &st) void Loader::load_sub_with(Loader &ldr) { - if(!cur_st) + if(direct) + { + ldr.load_direct(*cur_parser, cur_level+1); + ldr.finish(); + } + else if(cur_st) + { + ldr.load(*cur_st); + sub_loaded = true; + } + else throw logic_error("no current statement"); - - ldr.load(*cur_st); - sub_loaded = true; } void Loader::add(const string &kwd, LoaderAction *act) @@ -176,11 +191,22 @@ LoaderAction *Loader::find_action(const StatementKey &key) const if(begin==end) throw unknown_keyword(key.keyword); + LoaderAction *act = 0; + int match = 0; for(ActionMap::const_iterator i=begin; i!=end; ++i) - if(signature_match(key.signature, i->first.signature)) - return i->second; + { + int m = signature_match(key.signature, i->first.signature); + if(m>match) + { + act = i->second; + match = m; + } + } + + if(!act) + throw invalid_signature(key.keyword, key.signature); - throw invalid_signature(key.keyword, key.signature); + return act; } const string &Loader::get_source() const @@ -190,5 +216,12 @@ const string &Loader::get_source() const return cur_st->source; } +const string &Loader::get_keyword() const +{ + if(!cur_st) + throw logic_error("no current statement"); + return cur_st->keyword; +} + } // namespace DataFile } // namespace Msp