source: cpp/frams/util/extvalue.cpp @ 489

Last change on this file since 489 was 478, checked in by Maciej Komosinski, 9 years ago

Accessing const objects, short -> paInt, less critical messages when not necessary, accessing dictionaries with "->"

  • Property svn:eol-style set to native
File size: 29.1 KB
Line 
1// This file is a part of Framsticks SDK.  http://www.framsticks.com/
2// Copyright (C) 1999-2015  Maciej Komosinski and Szymon Ulatowski.
3// See LICENSE.txt for details.
4
5#include "extvalue.h"
6#include <frams/param/param.h>
7#include "sstringutils.h"
8#include <ctype.h>
9#include <frams/vm/classes/collectionobj.h>
10#include <frams/vm/classes/3dobject.h>
11#include <frams/vm/classes/genoobj.h>
12#include <common/nonstd_math.h>
13#include <common/Convert.h>
14#include <climits>
15#include <errno.h>
16
17#ifndef NO_BARRIER
18#include <frams/simul/barrier.h>
19#include <common/threads.h>
20#endif
21
22#ifdef MULTITHREADED
23#include <pthread.h>
24//this lock only protects against ref.counter corruption caused by concurrent reads.
25//read/write conficts and nonatomicity are handled by BarrierObject (at least in theory ;-))
26static pthread_mutex_t extobject_ref_lock = PTHREAD_MUTEX_INITIALIZER;
27#define REF_LOCK pthread_mutex_lock(&extobject_ref_lock)
28#define REF_UNLOCK pthread_mutex_unlock(&extobject_ref_lock)
29#else
30#define REF_LOCK
31#define REF_UNLOCK
32#endif
33
34void ExtObject::incref() const
35{
36        if (subtype & 1)
37        {
38                REF_LOCK;
39                dbobject->refcount++;
40                REF_UNLOCK;
41        }
42}
43
44void ExtObject::decref() const
45{
46        if (subtype & 1)
47        {
48                REF_LOCK;
49                bool destroy = !--dbobject->refcount;
50                REF_UNLOCK;
51                //another thread can now access the object while we are deleting it
52                //but this is not a bug since we only guarantee read/read safety
53                if (destroy) delete dbobject;
54        }
55}
56
57bool ExtObject::operator==(const ExtObject& src) const
58{
59        if (object != src.object) return false;
60        const char* n1 = interfaceName();
61        const char* n2 = src.interfaceName();
62        return (n1 == n2) || (strcmp(n1, n2) == 0);
63}
64
65bool ExtObject::makeUnique()
66{
67        if (!(subtype & 1)) return false;
68        if (dbobject->refcount == 1) return false;
69        VectorObject* v = VectorObject::fromObject(*this, false);
70        if (v)
71        {
72                VectorObject* n = new VectorObject;
73                n->data.setSize(n->data.size());
74                for (int i = 0; i < v->data.size(); i++)
75                {
76                        ExtValue *x = (ExtValue*)v->data(i);
77                        n->data.set(i, x ? new ExtValue(*x) : NULL);
78                }
79                operator=(n->makeObject());
80                return true;
81        }
82        return false;
83}
84
85void* ExtObject::getTarget(const char* classname, bool through_barrier, bool warn) const
86{
87        if (!strcmp(interfaceName(), classname))
88                return getTarget();
89#ifndef NO_BARRIER
90        if (through_barrier)
91        {
92                BarrierObject *bo = BarrierObject::fromObject(*this);
93                if (bo)
94                        return bo->getSourceObject().getTarget(classname, true, warn);
95        }
96#endif
97
98        if (warn)
99        {
100                logPrintf("ExtValue", "getObjectTarget", LOG_WARN, "%s object expected, %s found", classname, interfaceName());
101        }
102
103        return NULL;
104}
105
106bool ExtObject::callDelegate(const char* delegate,ExtValue *args,ExtValue *ret)
107{
108Param tmp;
109ParamInterface *pi=getParamInterface(tmp);
110if (pi)
111        {
112        int f=pi->findId(delegate);
113        if (f>=0)
114                {
115                pi->call(f,args,ret);
116                return true;
117                }
118        }
119logPrintf("Genotype","get", LOG_ERROR, "Could not call delegate '%s.%s'", pi?pi->getName():"NULL",delegate);
120return false;
121}
122
123SString ExtObject::toString() const
124{
125        if (isEmpty()) return SString("null");
126        Param tmp_param;
127        ParamInterface *p = getParamInterface(tmp_param);
128        int tostr = p->findId("toString");
129        if (tostr >= 0)
130        {
131                return SString(p->getString(tostr));
132        }
133        else
134        {
135                SString tmp("<");
136                tmp += p->getName();
137                tmp += SString::sprintf(" object at %p>", object ? object : paraminterface);
138                return tmp;
139        }
140}
141
142THREAD_LOCAL_DEF(ExtObject::Serialization, ExtObject_serialization);
143
144void ExtObject::Serialization::begin()
145{
146        if (level == 0)
147                refs.clear();
148        level++;
149}
150
151int ExtObject::Serialization::add(const ExtObject &o)
152{
153        if (o.isEmpty()) return -1;
154        for (int i = 0; i < (int)refs.size(); i++)
155        {
156                ExtObject& r = refs[i];
157                if (r == o) return i;
158        }
159        refs.push_back(o);
160        return -1;
161}
162
163void ExtObject::Serialization::replace(const ExtObject& o, const ExtObject& other)
164{
165        if (o.isEmpty()) return;
166        for (int i = 0; i < (int)refs.size(); i++)
167        {
168                ExtObject& r = refs[i];
169                if (r == o)
170                {
171                        r = other;
172                        return;
173                }
174        }
175}
176
177void ExtObject::Serialization::remove(const ExtObject& o)
178{
179        if (o.isEmpty()) return;
180        for (int i = 0; i < (int)refs.size(); i++)
181        {
182                ExtObject& r = refs[i];
183                if (o == r) refs.erase(refs.begin() + i);
184        }
185}
186
187const ExtObject* ExtObject::Serialization::get(int ref)
188{
189        if (ref < 0) return NULL;
190        if (ref >= (int)refs.size()) return NULL;
191        return &refs[ref];
192}
193
194void ExtObject::Serialization::end()
195{
196        level--;
197        if (level == 0)
198                refs.clear();
199}
200
201SString ExtObject::serialize_inner(SerializationFormat format) const
202{
203        int ref = tlsGetRef(ExtObject_serialization).add(*this);
204        SString ret;
205
206        if (ref >= 0)
207                {
208                switch(format)
209                        {
210                        case NativeSerialization: return SString::sprintf("^%d", ref);
211                        case JSONSerialization: return SString("null");
212                        }
213                }
214
215        if (isEmpty()) return SString("null");
216        {
217        VectorObject *vec = VectorObject::fromObject(*this, false);
218        if (vec)
219                { ret=vec->serialize(format); goto finally; }
220        }
221        {
222        DictionaryObject *dic = DictionaryObject::fromObject(*this, false);
223        if (dic)
224                { ret=dic->serialize(format); goto finally; }
225        }
226        {
227        Param tmp_param;
228        ParamInterface *p = getParamInterface(tmp_param);
229        int m = p->findId("toVector");
230        if (m < 0)
231                m = p->findId("toDictionary");
232        if (m >= 0)
233        {
234                ExtObject o(p->getObject(m));
235                switch(format)
236                        {
237                        case NativeSerialization: ret=SString(interfaceName())+o.serialize(format); break;
238                        case JSONSerialization: ret=SString::sprintf("{\"class\":\"%s\",\"data\":%s}",interfaceName(),o.serialize(format).c_str()); break;
239                        }
240                goto finally;
241        }
242        m = p->findId("toString");
243        if (m >= 0)
244        {
245                SString str = p->getString(m);
246                sstringQuote(str);
247                switch(format)
248                        {
249                        case NativeSerialization: ret=SString(interfaceName())+"\"" +str+"\""; break;
250                        case JSONSerialization: ret=SString::sprintf("{\"class\":\"%s\",\"data\":\"%s\"}",interfaceName(),str.c_str()); break;
251                        }
252                goto finally;
253        }
254        }
255
256        tlsGetRef(ExtObject_serialization).remove(*this);//undo nonserializable reference
257        switch(format)
258                {
259                case NativeSerialization: return SString(interfaceName())+SString::sprintf("<%p>",object ? object : paraminterface); break;
260                case JSONSerialization: return SString::sprintf("{\"class\":\"%s\"}",interfaceName()); break;
261                }
262
263  finally: // not 100% "finally", the case of nonserializable reference (directly above) returns directly without going through finally
264
265        switch(format)
266                {
267                case JSONSerialization:
268                        tlsGetRef(ExtObject_serialization).remove(*this);//JSON only tracks recursion, does not track reuse
269                        break;
270                case NativeSerialization:; //nop (just to avoid compiler warning)
271                }
272
273        return ret;
274}
275
276SString ExtObject::serialize(SerializationFormat format) const
277{
278        tlsGetRef(ExtObject_serialization).begin();
279        SString ret = serialize_inner(format);
280        tlsGetRef(ExtObject_serialization).end();
281        return ret;
282}
283
284///////////////////////////////////////
285
286SString ExtValue::typeDescription() const
287{
288        switch (type)
289        {
290        case TInt: return SString("int");
291        case TDouble: return SString("float");
292        case TString: return SString("string");
293        case TUnknown: return SString("null");
294        case TInvalid: return SString("invalid");
295        case TObj: return getObject().isEmpty() ? SString("null") : SString(getObject().interfaceName());
296        }
297        return SString::empty();
298}
299
300SString ExtValue::typeAndValue() const
301{
302        SString msg = typeDescription();
303        switch (type)
304        {
305        case TInt: case TDouble: case TString: case TObj:
306                msg += " '";
307                msg += getString();
308                msg += "'";
309        default:;
310        }
311        return msg;
312}
313
314void *ExtValue::getObjectTarget(const char* classname, bool warn) const
315{
316        if (type != TObj)
317        {
318                if (warn)
319                {
320                        SString tmp = getString();
321                        if (tmp.len() > 30) tmp = tmp.substr(0, 30) + "...";
322                        if (type == TString) tmp = SString("\"") + tmp + SString("\"");
323                        logPrintf("ExtValue", "getObjectTarget", LOG_WARN, "%s object expected, %s found", classname, tmp.c_str());
324                }
325                return NULL;
326        }
327
328        return getObject().getTarget(classname, true, warn);
329}
330
331void ExtValue::set(const ExtValue& src)
332{
333        switch (src.type)
334        {
335        case TString: sets(src.sdata()); break;
336        case TInt: seti(src.idata()); break;
337        case TDouble: setd(src.ddata()); break;
338        case TObj: seto(src.odata()); break;
339        default:type = src.type; break;
340        }
341}
342
343void ExtValue::setEmpty()
344{
345        switch (type)
346        {
347#ifdef EXTVALUEUNION
348        case TString: sdata().~SString(); break;
349        case TObj: odata().~ExtObject(); break;
350#else
351        case TString: delete s; break;
352        case TObj: delete o; break;
353#endif
354        default:;
355        }
356        type = TUnknown;
357}
358
359static ExtValue::CompareResult longsign(paInt x)
360{
361        if (x < 0) return ExtValue::ResultLower;
362        if (x > 0) return ExtValue::ResultHigher;
363        return ExtValue::ResultEqual;
364}
365
366static ExtValue::CompareResult compareNull(const ExtValue& v)
367{
368        if (v.isNull()) return ExtValue::ResultEqualUnordered;
369        if ((v.getType() == TInt) && (v.getInt() == 0)) return ExtValue::ResultUnequal_RelaxedEqual;
370        return ExtValue::ResultUnequal_RelaxedUnequal; //comparing anything else with null is valid but null is neither higher nor lower than numbers or strings
371}
372
373static ExtValue::CompareResult compareFloat(double a, double b)
374{
375        double t = a - b;
376        if (t < 0) return ExtValue::ResultLower;
377        else if (t > 0) return ExtValue::ResultHigher;
378        return ExtValue::ResultEqual;
379}
380
381static ExtValue::CompareResult compareString(const SString &a, const SString &b)
382{
383        const char* s1 = a.c_str();
384        const char* s2 = b.c_str();
385        return longsign(strcmp(s1, s2));
386}
387
388ExtValue::CompareResult ExtValue::compare(const ExtValue& src) const
389{
390        if (isNull())
391                return compareNull(src);
392        else if (src.isNull())
393                return compareNull(*this);
394        switch (type)
395        {
396
397        case TInt:
398
399                if (src.getType() == TInt)
400                {
401                        paInt t = src.getInt();
402                        if (idata() > 0)
403                        {
404                                if (t > 0) return longsign(idata() - t); else return ResultHigher;
405                        }
406                        else
407                        {
408                                if (t <= 0) return longsign(idata() - t); else return ResultLower;
409                        }
410                }
411                else if (src.getType() == TDouble)
412                        return compareFloat((double)idata(), src.getDouble());
413                else
414                        return ResultMismatch;//comparing numbers with other things is invalid
415                break;
416
417        case TDouble:
418                if ((src.getType() == TDouble) || (src.getType() == TInt))
419                        return compareFloat(getDouble(), src.getDouble());
420                else
421                        return ResultMismatch;
422                break;
423
424        case TString:
425                if (src.getType() == TString)
426                        return compareString(sdata(), src.getString());
427                else
428                        return ResultMismatch;
429                break;
430
431        case TObj:
432        {
433                if (src.type == TObj)
434                        return odata() == src.odata() ? ResultEqualUnordered : ResultUnequal_RelaxedUnequal;
435                if ((src.type == TInt) && (src.getInt() == 0))
436                        return ResultMismatch_RelaxedUnequal;
437                return ResultMismatch;
438        }
439        default:;
440        }
441        return ResultMismatch;
442}
443
444const char* ExtValue::cmp_op_names[] = { "==", "!=", ">=", "<=", ">", "<", "~=", "!~", NULL };
445
446int ExtValue::interpretCompare(CmpOperator op, CompareResult result, CmpContext *context)
447{
448        CompareResult error_threshold = ResultUnequal_RelaxedEqual;//error when ResultUnequal_RelaxedEqual or higher (not comparable)
449        int ret = 0;
450        switch (op)
451        {
452        case CmpEQ: ret = (result == ResultEqual) || (result == ResultEqualUnordered); error_threshold = ResultMismatch_RelaxedUnequal; break;
453        case CmpNE: ret = !((result == ResultEqual) || (result == ResultEqualUnordered)); error_threshold = ResultMismatch_RelaxedUnequal; break;
454        case CmpGT: ret = (result == ResultHigher); error_threshold = ResultEqualUnordered; break;
455        case CmpGE: ret = (result == ResultEqual) || (result == ResultHigher); error_threshold = ResultEqualUnordered; break;
456        case CmpLT: ret = (result == ResultLower); error_threshold = ResultEqualUnordered; break;
457        case CmpLE: ret = (result == ResultEqual) || (result == ResultLower); error_threshold = ResultEqualUnordered; break;
458        case CmpREQ: ret = (result == ResultEqual) || (result == ResultEqualUnordered) || (result == ResultUnequal_RelaxedEqual); error_threshold = ResultMismatch; break;
459        case CmpRNE: ret = !((result == ResultEqual) || (result == ResultEqualUnordered) || (result == ResultUnequal_RelaxedEqual)); error_threshold = ResultMismatch; break;
460        default:;
461        }
462        if (result >= error_threshold)
463        {
464                SString msg = "Type mismatch while comparing";
465                if (context)
466                {
467                        if (context->v1 && context->v2)
468                                msg += SString::sprintf(": %s %s %s",
469                                context->v1->typeAndValue().c_str(),
470                                cmp_op_names[op - CmpFIRST],
471                                context->v2->typeAndValue().c_str());
472                }
473                logPrintf("ExtValue", "interpretCompare", LOG_ERROR, "%s", msg.c_str());
474                ret = -1;
475        }
476        return ret;
477}
478
479int ExtValue::operator==(const ExtValue& src) const
480{
481        if (type != src.type) return 0;
482        switch (type)
483        {
484        case TInt: return idata() == src.idata();
485        case TDouble: return ddata() == src.ddata();
486        case TString: return sdata() == src.sdata();
487        case TObj: return odata() == src.odata();
488        default:;
489        }
490        return 1;
491}
492
493void ExtValue::operator+=(const ExtValue& src)
494{
495        // return = ok, break = fail
496        switch (type)
497        {
498        case TInt:
499                switch (src.getType())
500                {
501                case TDouble:
502                        setDouble(double(getInt()) + src.getDouble());
503                        return;
504                case TString:
505                        break;
506                default:
507                        idata() += src.getInt();
508                        return;
509                }
510                break;
511        case TDouble:
512                switch (src.getType())
513                {
514                case TString:
515                        break;
516                default:
517                        ddata() += src.getDouble();
518                        return;
519                }
520                break;
521        case TString: sdata() += src.getString(); return;
522        case TObj:
523        {
524                VectorObject *vec = VectorObject::fromObject(getObject(), false);
525                VectorObject *vec2 = VectorObject::fromObject(src.getObject(), false);
526                if (vec && vec2)
527                {
528                        for (int i = 0; i < vec2->data.size(); i++)
529                        {
530                                ExtValue *s = (ExtValue*)vec2->data(i);
531                                ExtValue *d = s ? new ExtValue(*s) : NULL;
532                                vec->data += d;
533                        }
534                        return;
535                }
536        }
537                //NO break;
538        default:;
539        }
540        logPrintf("ExtValue", "add", LOG_ERROR, "Can't add %s to %s", src.typeAndValue().c_str(), typeAndValue().c_str());
541}
542
543void ExtValue::operator-=(const ExtValue& src)
544{
545        // return = ok, break = fail
546        switch (type)
547        {
548        case TInt:
549                switch (src.getType())
550                {
551                case TInt:
552                        idata() -= src.getInt();
553                        return;
554                case TDouble:
555                        setDouble(double(getInt()) - src.getDouble());
556                        return;
557                default:;
558                }
559                break;
560        case TDouble:
561                switch (src.getType())
562                {
563                case TDouble:
564                case TInt:
565                        ddata() -= src.getDouble();
566                        return;
567                default:;
568                }
569                break;
570        default:;
571        }
572        logPrintf("ExtValue", "subtract", LOG_ERROR, "Can't subtract %s from %s", src.typeAndValue().c_str(), typeAndValue().c_str());
573}
574
575void ExtValue::operator*=(const ExtValue& src)
576{
577        // return = ok, break = fail
578        switch (type)
579        {
580        case TInt:
581                switch (src.getType())
582                {
583                case TInt:
584                        idata() *= src.getInt();
585                        return;
586                case TDouble:
587                        setDouble(double(getInt())*src.getDouble());
588                        return;
589                default:;
590                }
591                break;
592        case TDouble:
593                switch (src.getType())
594                {
595                case TInt:
596                case TDouble:
597                        ddata() *= src.getDouble();
598                        return;
599                default:;
600                }
601                break;
602        case TString:
603                switch (src.getType())
604                {
605                case TInt: case TDouble:
606                {
607                        SString t;
608                        for (int n = src.getInt(); n > 0; n--)
609                                t += getString();
610                        setString(t);
611                        return;
612                }
613                default:;
614                }
615                break;
616        case TObj:
617        {
618                VectorObject *vec = VectorObject::fromObject(getObject(), false);
619                if (vec)
620                {
621                        int n = src.getInt();
622                        int orig_size = vec->data.size();
623                        if (n <= 0)
624                        {
625                                vec->clear(); return;
626                        }
627                        for (; n > 1; n--)
628                        {
629                                for (int i = 0; i < orig_size; i++)
630                                {
631                                        ExtValue *s = (ExtValue*)vec->data(i);
632                                        ExtValue *d = s ? new ExtValue(*s) : NULL;
633                                        vec->data += d;
634                                }
635                        }
636                        return;
637                }
638        }
639                //NO break;
640        default:;
641        }
642        logPrintf("ExtValue", "multiply", LOG_WARN, "Can't multiply %s by %s", typeAndValue().c_str(), src.typeAndValue().c_str());
643}
644
645/*#include "fpu_control.h"
646#include <signal.h>
647
648static int fpuexception;
649void mathhandler(int sig)
650{
651printf("fpu exception!\n");
652fpuexception=1;
653signal(SIGFPE,SIG_IGN);
654} */
655
656void ExtValue::divInt(paInt a)
657{
658        if (a)
659                idata() /= a;
660        else
661        {
662                logPrintf("ExtValue", "divide", LOG_ERROR, "Division by zero: %d/0", idata());
663                setInvalid();
664        }
665}
666
667void ExtValue::divDouble(double a)
668{
669        if (a == 0.0)
670        {
671                logPrintf("ExtValue", "divide", LOG_ERROR, "Division by zero: %s/0.0", getString().c_str());
672                setInvalid();
673        }
674        else
675        {
676                fpExceptDisable();
677                double tmp = getDouble() / a;
678                if (!finite(tmp))
679                {
680                        logPrintf("ExtValue", "divide", LOG_ERROR, "Overflow %s/%g", getString().c_str(), a); setInvalid();
681                }
682                else
683                        setDouble(tmp);
684                // niby dobrze ale lepiej byloby to robic bardziej systematycznie a nie tylko w dzieleniu?
685                //if (isnan(ddata())) //http://www.digitalmars.com/d/archives/c++/Traping_divide_by_zero_5728.html
686                //        { logPrintf("ExtValue","divide",LOG_ERROR,"not-a-number",(const char*)getString()); setInvalid(); }
687                fpExceptEnable();
688        }
689}
690
691void ExtValue::operator/=(const ExtValue& src)
692{
693        switch (type)
694        {
695        case TInt:
696                switch (src.getType())
697                {
698                case TInt:
699                        divInt(src.idata());
700                        return;
701                case TDouble:
702                        divDouble(src.ddata());
703                        return;
704                default:;
705                }
706                break;
707
708        case TDouble:
709                switch (src.getType())
710                {
711                case TInt:
712                        divDouble(src.getDouble());
713                        return;
714                case TDouble:
715                        divDouble(src.ddata());
716                        return;
717                default:;
718                }
719                break;
720
721        default:;
722        }
723        logPrintf("ExtValue", "divide", LOG_ERROR, "Can't divide %s by %s", typeAndValue().c_str(), src.typeAndValue().c_str());
724}
725
726SString ExtValue::format(SString& fmt, const ExtValue **values, int count)
727{
728        SString ret;
729        // "..........%.........%..........%........"
730        //  ^_cur     ^_next
731        //  ^^^^^^^^^^___sub
732        //
733        // "..........%.........%..........%........"
734        //            ^-cur     ^-next
735        //            ^^^^^^^^^^___sub
736        const char* begin = fmt.c_str(), *end = begin + fmt.len(), *curr = begin;
737        int type = 0;
738
739        class Args
740        {
741                const ExtValue **values;
742                int count;
743                int arg;
744        public:
745                Args(const ExtValue **v, int c) :values(v), count(c), arg(0) {}
746                bool finished() { return arg >= count; }
747                const ExtValue *getNext() { const ExtValue *ret = NULL; if ((arg < count) && values[arg]) ret = values[arg]; arg++; return ret; }
748        };
749        Args args(values, count);
750
751        while (curr < end)
752        {
753                const char* next = strchr(curr, '%');
754                if (!next) next = end; else if ((next == curr) && (curr > begin))
755                {
756                        next = strchr(next + 1, '%'); if (!next) next = end;
757                }
758                type = 0;
759                if (curr > begin)
760                {
761                        type = 0;
762                        for (const char* t = curr; t < next; t++)
763                                switch (*t)
764                        {
765                                case 'd': case 'x': case 'X': case 'u': case 'p': case 'c': type = 'd'; t = next; break;
766                                case 'f': case 'g': case 'e': type = 'f'; t = next; break;
767                                case 's': type = 's'; t = next; break;
768                                case 't': case 'T': type = *t; t = next; break;
769                                case '%': if (t > begin) { type = *t; t = next; } break;
770                        }
771                }
772                if (curr > begin) curr--;
773                const ExtValue *a;
774                if (args.finished() && (type != 0) && (type != '%'))
775                {
776                        ret += fmt.substr((int)(curr - begin));
777                        break;
778                }
779                SString sub = fmt.substr((int)(curr - begin), (int)(next - curr));
780                switch (type)
781                {
782                case 'd': a = args.getNext(); ret += SString::sprintf(sub.c_str(), a ? a->getInt() : 0); break;
783                case 'f': a = args.getNext(); ret += SString::sprintf(sub.c_str(), a ? a->getDouble() : 0); break;
784                case 's': {a = args.getNext(); SString tmp; if (a) tmp = a->getString(); ret += SString::sprintf(sub.c_str(), tmp.c_str()); } break;
785                case 't': case 'T':
786                {
787                        a = args.getNext();
788                        time_t ti = a ? a->getInt() : 0;
789                        struct tm tm = Convert::localtime(ti);
790                        SString timtxt;
791                        if (type == 'T')
792                                timtxt = SString::sprintf("%04d-%02d-%02d %02d:%02d:%02d", 1900 + tm.tm_year, 1 + tm.tm_mon, tm.tm_mday, tm.tm_hour, tm.tm_min, tm.tm_sec);
793                        else
794                                timtxt = Convert::asctime(tm).c_str();
795                        ret += timtxt;
796                        ret += sub.substr(2);
797                }
798                        break;
799                case '%': ret += '%'; ret += sub.substr(2); break;
800                case 0: ret += sub; break;
801                }
802                curr = next + 1;
803        }
804        return ret;
805}
806
807
808void ExtValue::operator%=(const ExtValue& src)
809{
810        switch (type)
811        {
812        case TInt: idata() = idata() % src.getInt(); break;
813        case TDouble: ddata() = fmod(ddata(), src.getDouble()); break;
814
815        case TString:
816        {
817                VectorObject *vec = VectorObject::fromObject(src.getObject(), false);
818                if (vec)
819                        sdata() = format(sdata(), (const ExtValue**)&vec->data.getref(0), vec->data.size());
820                else
821                {
822                        const ExtValue *ptr = &src; sdata() = ExtValue::format(sdata(), &ptr, 1);
823                }
824        }
825                break;
826
827        case TObj: case TUnknown: case TInvalid:
828                logPrintf("ExtValue", "modulo", LOG_WARN, "Can't apply modulo to %s", typeDescription().c_str());
829
830        default:;
831        }
832}
833
834bool ExtValue::parseInt(const char* s, paInt &result, bool strict, bool error)
835{
836        ExtValue tmp;
837        const char* after = tmp.parseNumber(s, strict ? TInt : TUnknown);
838        if ((after == NULL) || (after[0] != 0))
839        {
840                if (error)
841                        logPrintf("ExtValue", "parseInt", LOG_ERROR, "Could not parse '%s'%s", s, strict ? " (strict)" : "");
842                return false;
843        }
844        result = tmp.getInt();
845        return true;
846}
847
848bool ExtValue::parseDouble(const char* s, double &result, bool error)
849{
850        ExtValue tmp;
851        const char* after = tmp.parseNumber(s, TDouble);
852        if ((after == NULL) || (after[0] != 0))
853        {
854                if (error)
855                        logPrintf("ExtValue", "parseDouble", LOG_ERROR, "Could not parse '%s'", s);
856                return false;
857        }
858        result = tmp.getDouble();
859        return true;
860}
861
862paInt ExtValue::getInt(const char* s, bool strict)
863{
864        paInt result;
865        if (parseInt(s, result, strict, true))
866                return result;
867        return 0;
868}
869
870double ExtValue::getDouble(const char* s)
871{
872        double result;
873        if (parseDouble(s, result, true))
874                return result;
875        return 0;
876}
877
878paInt ExtValue::getInt() const
879{
880        switch (type)
881        {
882        case TInt: return idata();
883        case TDouble: return (int)ddata();
884        case TString: return getInt(sdata().c_str());
885        case TObj:
886                logPrintf("ExtValue", "getInt", LOG_ERROR, "Getting integer value from object reference (%s)", getString().c_str());
887                return (paInt)(intptr_t)odata().param;
888        default:;
889        }
890        return 0;
891}
892
893double ExtValue::getDouble() const
894{
895        switch (type)
896        {
897        case TDouble: return ddata();
898        case TInt: return (double)idata();
899        case TString: return getDouble(sdata().c_str());
900        case TObj:
901                logPrintf("ExtValue", "getDouble", LOG_ERROR, "Getting floating point value from object reference (%s)", getString().c_str());
902                return (double)(intptr_t)odata().param;
903        default:;
904        }
905        return 0.0;
906}
907
908SString ExtValue::getString() const
909{
910        switch (type)
911        {
912        case TString: return sdata();
913        case TInt: return SString::valueOf(idata());
914        case TDouble: return SString::valueOf(ddata());
915        case TObj: return odata().toString();
916        case TInvalid:  return SString("invalid");
917        default: return SString("null");
918        }
919}
920
921const SString* ExtValue::getStringPtr() const
922{
923        if (type == TString)
924                return &sdata();
925        return NULL;
926}
927
928SString ExtValue::serialize(SerializationFormat format) const
929{
930        switch (type)
931        {
932        case TString:
933        {
934                SString q = sdata();
935                sstringQuote(q);
936                return SString("\"") + q + SString("\"");
937        }
938        case TInt:
939                return SString::valueOf(idata());
940        case TDouble:
941                return SString::valueOf(ddata());
942        case TObj:
943                return odata().serialize(format);
944        case TInvalid:
945                if (format==NativeSerialization)
946                        return SString("invalid");
947                // else null --v
948        default:
949                return SString("null");
950        }
951}
952
953/// returns the first character after the parsed number, or NULL if not a number
954/// @param strict_type = restrict the allowed return value (TUnknown = unrestricted)
955const char* ExtValue::parseNumber(const char* in, ExtPType strict_type)
956{
957        char* after;
958        if (in == NULL) return NULL;
959        if (in[0] == 0) return NULL;
960        while (isspace(*in)) in++;
961        bool minus = (in[0] == '-');
962        bool plus = (in[0] == '+');
963        if (((in[0] == '0') && ((in[1] == 'x') || (in[1] == 'X')))
964                || (((minus || plus) && (in[1] == '0') && ((in[2] == 'x') || (in[2] == 'X')))))
965        {
966                in += (minus || plus) ? 3 : 2;
967                if (isspace(*in)) return NULL;
968                errno = 0;
969                unsigned long intvalue = strtoul(in, &after, 16);
970                if ((after > in) && (errno == 0) && (intvalue <= 0xffffffff))
971                {
972                        if (strict_type == TDouble)
973                                setDouble(minus ? -(double)intvalue : (double)intvalue);
974                        else
975                                setInt(minus ? -intvalue : intvalue);
976                        return after;
977                }
978                else
979                        return NULL;
980        }
981
982        errno = 0;
983        double fpvalue = strtod(in, &after);
984        if ((after > in) && (errno == 0))
985        {
986                if (strict_type != TDouble)
987                {
988                        if ((memchr(in, '.', after - in) == NULL) && (memchr(in, 'e', after - in) == NULL) && (memchr(in, 'E', after - in) == NULL) // no "special" characters
989                                && (fpvalue == floor(fpvalue)) // value is integer
990                                && (fpvalue >= INT_MIN) && (fpvalue <= INT_MAX)) // within limits
991                        {
992                                setInt(fpvalue);
993                                return after;
994                        }
995                        else if (strict_type == TInt)
996                                return NULL;
997                }
998                setDouble(fpvalue);
999                return after;
1000        }
1001        return NULL;
1002}
1003
1004PtrListTempl<ParamInterface*> &ExtValue::getDeserializableClasses()
1005{
1006        static PtrListTempl<ParamInterface*> classes;
1007        return classes;
1008}
1009
1010ParamInterface *ExtValue::findDeserializableClass(const char* name)
1011{
1012        FOREACH(ParamInterface*, cls, getDeserializableClasses())
1013                if (!strcmp(cls->getName(), name))
1014                        return cls;
1015        return NULL;
1016}
1017
1018static const char* skipWord(const char* in)
1019{
1020        while (isalpha(*in) || (*in == '_'))
1021                in++;
1022        return in;
1023}
1024
1025//returns the first character after the parsed portion or NULL if invalid format
1026const char* ExtValue::deserialize_inner(const char* in)
1027{
1028        while(isspace(*in)) in++;
1029        const char* ret = parseNumber(in);
1030        if (ret)
1031                return ret;
1032        else if (*in == '\"')
1033        {
1034                ret = skipQuoteString(in + 1, NULL);
1035                SString s(in + 1, (int)(ret - (in + 1)));
1036                sstringUnquote(s);
1037                setString(s);
1038                if (*ret == '\"')
1039                        return ret + 1;
1040                else
1041                {
1042                        logPrintf("ExtValue", "deserialize", LOG_ERROR, "Missing '\"' in string: '%s'", ret);
1043                        return NULL;
1044                }
1045        }
1046        else if (*in == '[')
1047        {
1048                VectorObject *vec = new VectorObject;
1049                ExtObject o(&VectorObject::par, vec);
1050                tlsGetRef(ExtObject_serialization).add(o);
1051                const char* p = in + 1;
1052                ExtValue tmp;
1053                while (*p)
1054                {
1055                        if (isspace(*p)) p++;
1056                        if (*p == ']') { p++; break; }
1057                        ret = tmp.deserialize(p);
1058                        if (ret)
1059                        {
1060                                vec->data += new ExtValue(tmp);
1061                                p = ret;
1062                                if (*p == ',') p++;
1063                                else if (*p != ']')
1064                                {
1065                                        logPrintf("ExtValue", "deserialize", LOG_ERROR, "Missing ',' in Vector: '%s'", p);
1066                                        return NULL;
1067                                }
1068                        }
1069                        else
1070                        {
1071                                p = NULL;
1072                                break;
1073                        }
1074                }
1075                setObject(o);
1076                return p;
1077        }
1078        else if (*in == '{')
1079        {
1080                DictionaryObject *dic = new DictionaryObject;
1081                ExtObject o(&DictionaryObject::par, dic);
1082                tlsGetRef(ExtObject_serialization).add(o);
1083                const char* p = in + 1;
1084                ExtValue args[2]/*={value,key}*/, dummy_ret;
1085                while (*p)
1086                {
1087                        if (isspace(*p)) p++;
1088                        if (*p == '}') { p++; break; }
1089                        ret = args[1].deserialize(p);
1090                        if ((!ret) || (args[1].getType() != TString)) { p = NULL; break; }
1091                        p = ret;
1092                        if (*p != ':') { logPrintf("ExtValue", "deserialize", LOG_ERROR, "Missing ':' in Dictionary: '%s'", p); p = NULL; break; }
1093                        p++;
1094                        ret = args[0].deserialize(p);
1095                        if (!ret) { p = NULL; break; }
1096                        p = ret;
1097                        dic->p_set(args, &dummy_ret);
1098                        if (*p == ',') p++;
1099                        else if (*p != '}')
1100                        {
1101                                logPrintf("ExtValue", "deserialize", LOG_ERROR, "Missing ',' in Dictionary: '%s'", p);
1102                                return NULL;
1103                        }
1104                }
1105                setObject(o);
1106                return p;
1107        }
1108        else if (!strncmp(in, "null", 4))
1109        {
1110                setEmpty();
1111                return in + 4;
1112        }
1113        else if (!strncmp(in, "true", 4))
1114        {
1115                setInt(1);
1116                return in + 4;
1117        }
1118        else if (!strncmp(in, "false", 5))
1119        {
1120                setInt(0);
1121                return in + 5;
1122        }
1123        else if (!strncmp(in, "invalid", 9))
1124        {
1125                setInvalid();
1126                return in + 9;
1127        }
1128        else if (*in == '<')
1129        { //unserializable object
1130                setInvalid();
1131                while (*in)
1132                        if (*in == '>')
1133                                return in + 1;
1134                        else in++;
1135                        return in;
1136        }
1137        else if (*in == '^')
1138        {
1139                in++;
1140                ExtValue ref;
1141                ret = ref.parseNumber(in, TInt);
1142                if (ret && (ref.getType() == TInt))
1143                {
1144                        const ExtObject* o = tlsGetRef(ExtObject_serialization).get(ref.getInt());
1145                        if (o)
1146                        {
1147                                setObject(*o);
1148                                return ret;
1149                        }
1150                }
1151                logPrintf("ExtValue", "deserialize", LOG_ERROR, "Invalid reference: '%s'", in - 1);
1152                return NULL;
1153        }
1154        else if ((ret = skipWord(in)) && (ret != in))
1155        {
1156                SString clsname(in, (int)(ret - in));
1157                ExtValue tmp;
1158                ret = tmp.deserialize(ret);
1159                ParamInterface *cls = findDeserializableClass(clsname.c_str());
1160                if (cls && (tmp.getType() != TUnknown) && (tmp.getType() != TInvalid))
1161                {
1162                        VectorObject *vec = VectorObject::fromObject(tmp.getObject(), false);
1163                        if (vec)
1164                        {
1165                                int m = cls->findId("newFromVector");
1166                                if (m >= 0)
1167                                {
1168                                        cls->call(m, &tmp, this);
1169                                        tlsGetRef(ExtObject_serialization).replace(tmp.getObject(), getObject());
1170                                        return ret;
1171                                }
1172                        }
1173                        DictionaryObject *dic = DictionaryObject::fromObject(tmp.getObject(), false);
1174                        if (dic)
1175                        {
1176                                int m = cls->findId("newFromDictionary");
1177                                if (m >= 0)
1178                                {
1179                                        cls->call(m, &tmp, this);
1180                                        tlsGetRef(ExtObject_serialization).replace(tmp.getObject(), getObject());
1181                                        return ret;
1182                                }
1183                        }
1184                        if (tmp.getType() == TString)
1185                        {
1186                                int m = cls->findId("newFromString");
1187                                if (m >= 0)
1188                                {
1189                                        cls->call(m, &tmp, this);
1190                                        tlsGetRef(ExtObject_serialization).replace(tmp.getObject(), getObject());
1191                                        return ret;
1192                                }
1193                        }
1194                        tlsGetRef(ExtObject_serialization).remove(tmp.getObject());
1195                        setEmpty();
1196                }
1197                setEmpty();
1198                logPrintf("ExtValue", "deserialize", LOG_WARN, "object of class \"%s\" could not be deserialized", clsname.c_str());
1199                return ret;
1200        }
1201        logPrintf("ExtValue", "deserialize", LOG_ERROR, "Bad syntax: '%s'", in);
1202        setEmpty();
1203        return NULL;
1204}
1205
1206const char* ExtValue::deserialize(const char* in)
1207{
1208        tlsGetRef(ExtObject_serialization).begin();
1209        const char* ret = deserialize_inner(in);
1210        if (ret) while(isspace(*ret)) ret++;
1211        tlsGetRef(ExtObject_serialization).end();
1212        return ret;
1213}
1214
1215ExtObject ExtValue::getObject() const
1216{
1217        if (type == TObj) return odata();
1218        return ExtObject();
1219}
1220
1221ExtValue ExtValue::getExtType()
1222{
1223        static const char* typenames[] = { "null", "int", "float", "string", "", "invalid" };
1224        if (getType() != TObj)
1225                return ExtValue(typenames[(int)getType()]);
1226        ExtObject& o = odata();
1227        return ExtValue(SString(o.isEmpty() ? "" : o.interfaceName()));
1228}
1229
1230SString SString::valueOf(const ExtValue& v)
1231{
1232        return v.getString();
1233}
1234SString SString::valueOf(const ExtObject& v)
1235{
1236        return v.toString();
1237}
Note: See TracBrowser for help on using the repository browser.