diff options
Diffstat (limited to 'mcop/anyref.cpp')
-rw-r--r-- | mcop/anyref.cpp | 310 |
1 files changed, 310 insertions, 0 deletions
diff --git a/mcop/anyref.cpp b/mcop/anyref.cpp new file mode 100644 index 0000000..c5cf2cf --- /dev/null +++ b/mcop/anyref.cpp @@ -0,0 +1,310 @@ + /* + + Copyright (C) 2000 Stefan Westerfeld + + This library is free software; you can redistribute it and/or + modify it under the terms of the GNU Library General Public + License as published by the Free Software Foundation; either + version 2 of the License, or (at your option) any later version. + + This library is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU + Library General Public License for more details. + + You should have received a copy of the GNU Library General Public License + along with this library; see the file COPYING.LIB. If not, write to + the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, + Boston, MA 02110-1301, USA. + + */ + +#include "anyref.h" +#include "common.h" +#include "startupmanager.h" +#include "debug.h" +#include <assert.h> + +using namespace Arts; +using namespace std; + +/* + * private utility class to deal with any types + */ +namespace Arts { + static class AnyRefHelper { + private: + Arts::InterfaceRepoV2 interfaceRepo; + + public: + AnyRefHelper() { + interfaceRepo = DynamicCast(Dispatcher::the()->interfaceRepo()); + } + void skipType(Buffer& buffer, const string& type); + } *anyRefHelper = 0; + + static class AnyRefHelperStartup : public StartupClass { + void startup() { + assert(anyRefHelper == 0); + anyRefHelper = new AnyRefHelper(); + } + void shutdown() { + delete anyRefHelper; + anyRefHelper = 0; + } + } The_AnyRefHelperStartup; +} + +string AnyRefBase::type() const +{ +/* + keep common types here, so that the string constructor doesn't need + to get called each and every time + */ + static const string tVoid("void"); + static const string tByte("byte"); + static const string tsByte("*byte"); + static const string tLong("long"); + static const string tsLong("*long"); + static const string tFloat("float"); + static const string tsFloat("*float"); + static const string tString("string"); + static const string tsString("*string"); + static const string tBool("bool"); + static const string tsBool("*bool"); + + switch(rep) + { + case repVoid: return tVoid; + + // primitive types + case repByte: return tByte; + case repInt: + case repLong: return tLong; + case repFloat: + case repDouble: return tFloat; + case repString: + case repCString: return tString; + case repBool: return tBool; + + // sequences of primitive types + case repByteSeq: return tsByte; + case repLongSeq: return tsLong; + case repFloatSeq: return tsFloat; + case repStringSeq: return tsString; + case repBoolSeq: return tsBool; + + case repAny: return ((Any *)data)->type; + } + assert(false); + return tVoid; // silence compiler +} + +void AnyRefBase::_write(Buffer *b) const +{ + switch(rep) + { + case repVoid: + break; + + // primitive types + case repByte: b->writeByte(*(mcopbyte *)data); + break; + + case repInt: b->writeLong(*(int *)data); + break; + + case repLong: b->writeLong(*(long *)data); + break; + + case repFloat: b->writeFloat(*(float *)data); + break; + + case repDouble: b->writeFloat(*(double *)data); + break; + + case repString: b->writeString(*(string *)data); + break; + + case repCString: b->writeString((const char *)data); + break; + + case repBool: b->writeBool(*(bool *)data); + break; + + // sequences of primitive types + case repByteSeq: b->writeByteSeq(*(vector<mcopbyte> *)data); + break; + + case repLongSeq: b->writeLongSeq(*(vector<long> *)data); + break; + + case repFloatSeq: b->writeFloatSeq(*(vector<float> *)data); + break; + + case repStringSeq: b->writeStringSeq(*(vector<string> *)data); + break; + + case repBoolSeq: b->writeBoolSeq(*(vector<bool> *)data); + break; + + case repAny: b->write(((Any *)data)->value); + break; + + default: assert(false); + } +} + +void AnyRefBase::_read(Buffer *b) const +{ + switch(rep) + { + case repVoid: + break; + + // primitive types + case repByte: *(mcopbyte *)data = b->readByte(); + break; + + case repInt: *(int *)data = b->readLong(); + break; + + case repLong: *(long *)data = b->readLong(); + break; + + case repFloat: *(float *)data = b->readFloat(); + break; + + case repDouble: *(double *)data = b->readFloat(); + break; + + case repString: b->readString(*(string *)data); + break; + + case repBool: *(bool *)data = b->readBool(); + break; + + // sequences of primitive types + case repByteSeq: b->readByteSeq(*(vector<mcopbyte> *)data); + break; + + case repLongSeq: b->readLongSeq(*(vector<long> *)data); + break; + + case repFloatSeq: b->readFloatSeq(*(vector<float> *)data); + break; + + case repStringSeq: b->readStringSeq(*(vector<string> *)data); + break; + + case repBoolSeq: b->readBoolSeq(*(vector<bool> *)data); + break; + + case repAny: + { + // find out the size by skipping over it for the first time + long startPos = b->size() - b->remaining(); + anyRefHelper->skipType(*b, ((Any *)data)->type); + + // if everything went well, read the raw value in one step + long size = (b->size() - b->remaining()) - startPos; + if(!b->readError()) + { + b->rewind(); + b->skip(startPos); + b->read(((Any *)data)->value, size); + } + } + break; + + default: assert(false); + } +} + +/** + * correct skipping of an arbitary type not known at compile-time (this is + * a problem, since the size of the type will vary, due to the sequence<...>s + * contained) + */ +void AnyRefHelper::skipType(Buffer& buffer, const string& type) +{ + /* sequences */ + if(type[0] == '*') + { + long seqlen = buffer.readLong(); + while(seqlen > 0 && !buffer.readError()) + { + skipType(buffer, type.c_str()+1); + seqlen--; + } + } + else + { + TypeIdentification ti = interfaceRepo.identifyType(type); + switch(ti) + { + case tiString: + { + string s; + buffer.readString(s); + } + break; + + case tiLong: + buffer.readLong(); + break; + + case tiFloat: + buffer.readFloat(); + break; + + case tiByte: + buffer.readByte(); + break; + + case tiBoolean: + buffer.readBool(); + break; + + case tiVoid: + /* nothing to do */ + break; + + case tiType: + { + Arts::TypeDef td = interfaceRepo.queryType(type); + + if(td.name == type) + { + vector<TypeComponent>::iterator tci; + for(tci = td.contents.begin(); tci != td.contents.end(); + tci++) + { + skipType(buffer,tci->type); + } + } + else + { + arts_warning("unknown type %s",type.c_str()); + } + } + break; + + case tiEnum: + buffer.readLong(); + break; + + case tiInterface: + { + ObjectReference oref; + oref.readType(buffer); + } + break; + + default: + arts_warning("AnyRefHelper: can't read %s",type.c_str()); + break; + } + } +} |