source: cpp/frams/util/extvalue.h @ 117

Last change on this file since 117 was 109, checked in by sz, 11 years ago

source reorganization (see README)
new feature added: part/joint shapes (see frams/_demos/part_shapes.cpp)

  • Property svn:eol-style set to native
File size: 6.7 KB
Line 
1// This file is a part of the Framsticks GDK library.
2// Copyright (C) 2002-2011  Szymon Ulatowski.  See LICENSE.txt for details.
3// Refer to http://www.framsticks.com/ for further information.
4
5#ifndef _EXTVALUE_H_
6#define _EXTVALUE_H_
7
8#include "sstring.h"
9#include <frams/param/param.h>
10#include <common/nonstd_stl.h>
11#include <common/threads.h>
12
13#define EXTVALUEUNION
14template <int A,int B> struct CompileTimeMax {enum {val=A>B?A:B}; };
15#define EXTVALUEUNIONSIZE CompileTimeMax<sizeof(ExtObject),sizeof(SString)>::val
16
17enum ExtPType
18{TUnknown=0,TInt,TDouble,TString,TObj,TInvalid};
19
20/**
21   destructable object
22 */
23class DestrBase
24{
25public:
26int refcount;
27DestrBase():refcount(0) {}
28void incref() {refcount++;}
29void decref() {refcount--; if (refcount==0) delete this;}
30virtual ~DestrBase() {}
31};
32
33/**
34   object reference.
35 */
36class ExtObject
37{
38int subtype;                    //< 0/1=Generic/DPC Object,  0/2=Standalone/Shared Param
39void incref() const;
40void decref() const;
41  public:
42union { void* object;           //< generic object, will use param
43DestrBase *dbobject;};  //< object with refcounting, will be deleted if refcount goes to 0
44union { Param* param;           //< if object!=0
45        ParamInterface *paraminterface;}; //< if object==0
46
47void copyFrom(const ExtObject& src)  {subtype=src.subtype;object=src.object;param=src.param;}
48
49void* operator new(size_t s, void* mem) {return mem;}
50#ifdef _MSC_VER
51void operator delete(void* mem,void* t) {}
52#endif
53void* operator new(size_t s) {return malloc(sizeof(ExtObject));}
54void operator delete(void* mem) {free(mem);}
55///@param tmp_param can be used for temporary storage, the result ParamInterface* is only valid for as long as tmp_param is valid
56ParamInterface *getParamInterface(Param &tmp_param) const  {if(subtype&2){tmp_param.setParamTab(param->getParamTab());tmp_param.select(object);return &tmp_param;} return paraminterface;}
57const char* interfaceName() const {if (isEmpty()) return "Empty"; return (subtype&2)?param->getName():paraminterface->getName();}
58bool matchesInterfaceName(ParamInterface* pi) const {return !strcmp(interfaceName(),pi->getName());}
59void* getTarget() const {return (subtype&1)?dbobject:object;}
60void* getTarget(const char* classname, bool through_barrier=true) const;
61void setEmpty() {decref();subtype=0;param=NULL;object=NULL;}
62int isEmpty() const {return !param;}
63ExtObject(const ExtObject& src)      {src.incref();copyFrom(src);}
64void operator=(const ExtObject& src) {src.incref();decref();copyFrom(src);}
65bool makeUnique();//< @return false if nothing has changed
66
67int operator==(const ExtObject& src) const {if (object!=src.object) return 0; return (object==0)?(!strcmp(param->getName(),src.param->getName())):1;}
68
69SString toString() const;
70SString serialize_inner() const;
71SString serialize() const;
72
73ExtObject(Param *p,void *o):subtype(2),object(o),param(p){}
74ExtObject(ParamInterface *p=0):subtype(0),object(0),paraminterface(p){}
75ExtObject(Param *p,DestrBase *o):subtype(1+2),dbobject(o),param(p){incref();}
76ExtObject(ParamInterface *p,DestrBase *o):subtype(1),dbobject(o),paraminterface(p){incref();}
77
78~ExtObject(){decref();}
79
80class Serialization
81{
82std::vector<ExtObject> refs;
83int level;
84  public:
85Serialization():level(0) {}
86void begin();
87void end();
88int add(const ExtObject& o);
89void replace(const ExtObject& o,const ExtObject& other);
90void remove(const ExtObject& o);
91const ExtObject* get(int ref);
92};
93
94static THREAD_LOCAL_DECL(Serialization,serialization);
95};
96
97class ExtValue
98{
99public:
100ExtPType type;
101#ifdef EXTVALUEUNION
102long data[(EXTVALUEUNIONSIZE+sizeof(long)-1)/sizeof(long)];
103long& idata() const {return (long&)data[0];};
104double& ddata() const {return *(double*)data;};
105ExtObject& odata() const {return *(ExtObject*)data;};
106SString& sdata() const {return *(SString*)data;};
107#else
108union {
109long i;
110double d;
111SString *s;
112ExtObject *o;
113};
114long& idata() const {return (long&)i;};
115double& ddata() const {return (double&)d;};
116ExtObject& odata() const {return *o;};
117SString& sdata() const {return *s;};
118#endif
119
120void* operator new(size_t s, void* mem) {return mem;}
121void* operator new(size_t s) {return ::operator new(s);}
122
123ExtValue():type(TUnknown){}
124~ExtValue() {setEmpty();}
125ExtValue(long v) {seti(v);}
126ExtValue(double v) {setd(v);}
127ExtValue(const SString &v) {sets(v);}
128ExtValue(const ExtObject &srco) {seto(srco);}
129long compare(const ExtValue& src) const;
130int operator==(const ExtValue& src) const;
131void operator+=(const ExtValue& src);
132void operator-=(const ExtValue& src);
133void operator*=(const ExtValue& src);
134void operator/=(const ExtValue& src);
135void operator%=(const ExtValue& src);
136void operator=(const ExtValue& src)
137        {setr(src);}
138ExtValue(const ExtValue& src)
139        :type(TUnknown) {set(src);}
140void setEmpty();
141void setInvalid() {setEmpty();type=TInvalid;}
142bool makeUnique() {return (type==TObj) && odata().makeUnique();} //< @return false if nothing has changed
143ExtPType getType() {return type;}
144void *getObjectTarget(const char* classname) {return (type==TObj)?getObject().getTarget(classname):0;}
145void setInt(long v) {if (type!=TInt) setri(v); else idata()=v;}
146void setDouble(double v) {if (type!=TDouble) setrd(v); else ddata()=v;}
147void setString(const SString &v) {if (type!=TString) setrs(v); else sdata()=v;}
148void setObject(const ExtObject &src) {if (type!=TObj) setro(src); else odata()=src;}
149long getInt() const;
150double getDouble() const;
151SString getString() const;
152const SString* getStringPtr() const;//< @return pointer to the internal sstring object or NULL if the current type is not string
153SString serialize() const;
154ExtObject getObject() const;
155const char* parseNumber(const char* in);
156const char* deserialize(const char* in);//< @return first character after the succesfully parsed string or NULL if failed
157const char* deserialize_inner(const char* in);
158static PtrListTempl<ParamInterface*> deserializable_classes;
159static ParamInterface *findDeserializableClass(const char* name);
160static void initDeserializableClasses();
161static SString format(SString& fmt,const ExtValue **values,int count);
162
163ExtValue getExtType();
164
165  private: // setrx - release and set, setx - assume released
166void setr(const ExtValue& src){setEmpty();set(src);}
167void set(const ExtValue& src);
168void setri(long v) {setEmpty();seti(v);}
169void setrd(double v) {setEmpty();setd(v);}
170void seti(long v) {type=TInt;idata()=v;}
171void setd(double v) {type=TDouble;ddata()=v;}
172#ifdef EXTVALUEUNION
173void setrs(const SString &v) {setEmpty();sets(v);}
174void setro(const ExtObject &src) {setEmpty();seto(src);}
175void sets(const SString &v) {type=TString;new(data) SString(v);}
176void seto(const ExtObject &src) {type=TObj;new(data) ExtObject(src);}
177#else
178void setrs(const SString &v) {setEmpty();sets(v);}
179void setro(const ExtObject &src) {setEmpty();seto(src);}
180void sets(const SString &v) {type=TString;s=new SString(v);}
181void seto(const ExtObject &src) {type=TObj;o=new ExtObject(src);}
182#endif
183
184};
185
186
187#endif
Note: See TracBrowser for help on using the repository browser.