ecere/com/dataTypes: Avoid infinite recursion on bit classes holding themselves
[sdk] / ecere / src / com / dataTypes.ec
1 namespace com;
2
3 #if defined(ECERE_BOOTSTRAP)
4 #define dllexport
5 #endif
6
7 import "instance"
8
9 default extern Platform runtimePlatform;
10
11 // Limit Definitions
12 public define MAXBYTE = 0xff;
13 public define MAXWORD = 0xffff;
14 public define MININT = ((int)0x80000000);
15 public define MAXINT = ((int)0x7fffffff);
16 public define MININT64 = ((int64)0x8000000000000000LL);
17 public define MAXINT64 = ((int64)0x7fffffffffffffffLL);
18 public define MAXDWORD = 0xffffffff;
19 public define MAXQWORD = 0xffffffffffffffffLL;
20 public define MINFLOAT = 1.17549435082228750e-38f;
21 public define MAXFLOAT = 3.40282346638528860e+38f;
22 public define MINDOUBLE = 2.2250738585072014e-308;
23 public define MAXDOUBLE = 1.7976931348623158e+308;
24
25 public define FORMAT64HEXLL  = (__runtimePlatform == win32) ? "0x%I64XLL" : "0x%llXLL";
26 public define FORMAT64HEX    = (__runtimePlatform == win32) ? "0x%I64X" : "0x%llX";
27 public define FORMAT64DLL    = (__runtimePlatform == win32) ? "%I64dLL" : "%lldLL";
28 public define FORMAT64D      = (__runtimePlatform == win32) ? "%I64d" : "%lld";
29 public define FORMAT64U      = (__runtimePlatform == win32) ? "%I64u" : "%llu";
30
31 #define PUTXWORD(b, w) \
32    (b)[0] = (byte)(((w) >> 8) & 0xFF); \
33    (b)[1] = (byte)(((w)     ) & 0xFF);
34
35 #define GETXWORD(b) (uint16)(((b)[0] << 8) | (b)[1])
36
37 #define PUTXDWORD(b, d) \
38    (b)[0] = (byte)(((d) >> 24) & 0xFF); \
39    (b)[1] = (byte)(((d) >> 16) & 0xFF); \
40    (b)[2] = (byte)(((d) >> 8)  & 0xFF); \
41    (b)[3] = (byte)( (d)        & 0xFF);
42
43 #define GETXDWORD(b) (uint32)(((b)[0] << 24) | ((b)[1] << 16) | ((b)[2] << 8) | (b)[3])
44
45 #define PUTXQWORD(b, d) \
46    (b)[0] = (byte)(((d) >> 56) & 0xFF); \
47    (b)[1] = (byte)(((d) >> 48) & 0xFF); \
48    (b)[2] = (byte)(((d) >> 40) & 0xFF); \
49    (b)[3] = (byte)(((d) >> 32) & 0xFF); \
50    (b)[4] = (byte)(((d) >> 24) & 0xFF); \
51    (b)[5] = (byte)(((d) >> 16) & 0xFF); \
52    (b)[6] = (byte)(((d) >> 8)  & 0xFF); \
53    (b)[7] = (byte)( (d)        & 0xFF);
54
55 #define GETXQWORD(b) (uint64)(((uint64)(b)[0] << 56) | ((uint64)(b)[1] << 48) | ((uint64)(b)[2] << 40) | ((uint64)(b)[3] << 32) | ((uint64)(b)[4] << 24) | ((b)[5] << 16) | ((b)[6] << 8) | (b)[7])
56
57 __attribute__((unused)) static void UnusedFunction()
58 {
59    int a;
60    a.OnGetString(0,0,0);
61    a.OnFree();
62    a.OnCopy(0);
63    a.OnCompare(0);
64    a.OnSaveEdit(null,0);
65    a.OnEdit(null,null,0,0,0,20,0);
66    a.OnGetDataFromString(null);
67 }
68
69 #define uint _uint
70 #include <stdarg.h>
71 #include <stdio.h>
72 #undef uint
73
74 default:
75 FILE *eC_stdout(void);
76 FILE *eC_stderr(void);
77
78 bool Float_isNan(float n);
79 bool Float_isInf(float n);
80 int Float_signBit(float n);
81 float Float_nan(void);
82 float Float_inf(void);
83
84 bool Double_isNan(double n);
85 bool Double_isInf(double n);
86 int Double_signBit(double n);
87 double Double_nan(void);
88 double Double_inf(void);
89
90 extern int __ecereVMethodID_class_OnGetString;
91 extern int __ecereVMethodID_class_OnGetDataFromString;
92 extern int __ecereVMethodID_class_OnCompare;
93 extern int __ecereVMethodID_class_OnSerialize;
94 extern int __ecereVMethodID_class_OnUnserialize;
95 extern int __ecereVMethodID_class_OnCopy;
96 public:
97
98 #if defined(ECERE_BOOTSTRAP) || defined(ECERE_STATIC)
99 #define dllexport
100 #endif
101
102 public class IOChannel
103 {
104 public:
105    virtual uint WriteData(const void * data, uint numBytes);
106    virtual uint ReadData(void * data, uint numBytes);
107
108    dllexport void Serialize(typed_object data)
109    {
110       data.OnSerialize(this);
111    }
112
113    dllexport void Unserialize(typed_object & data)
114    {
115       data.OnUnserialize(this);
116    }
117
118    dllexport void Put(typed_object data)
119    {
120       data.OnSerialize(this);
121    }
122
123    dllexport void Get(typed_object & data)
124    {
125       data.OnUnserialize(this);
126    }
127 };
128
129 public class SerialBuffer : IOChannel
130 {
131 public:
132    byte * _buffer;
133    uint count;
134    uint _size;
135    uint pos;
136
137    uint WriteData(const void * bytes, uint numBytes)
138    {
139       if(this != null)
140       {
141          if(count + numBytes > _size)
142          {
143             _size = count + numBytes;
144             _size += _size/2;
145             _buffer = renew _buffer byte[_size];
146          }
147          memcpy(_buffer + count, bytes, numBytes);
148          count += numBytes;
149          return numBytes;
150       }
151       return 0;
152    }
153
154    uint ReadData(void * bytes, uint numBytes)
155    {
156       if(this != null)
157       {
158          int read = Min(numBytes, count - pos);
159          memcpy(bytes, _buffer + pos, read);
160          pos += read;
161          return read;
162       }
163       return 0;
164    }
165
166    ~SerialBuffer()
167    {
168       Free();
169    }
170
171    dllexport void Free()
172    {
173       if(this)
174       {
175          delete _buffer;
176          count = 0;
177          _size = 0;
178          pos = 0;
179       }
180    }
181
182    // TODO: THIS IS VERY BAD!
183    property byte * buffer
184    {
185       get { return _buffer + pos; }
186       set { _buffer = value; }
187    }
188
189    property uint size
190    {
191       get { return count - pos; }
192       set { count = value; }
193    }
194 };
195
196 /*static */const char * Enum_OnGetString(Class _class, void * data, char * tempString, void * fieldData, bool * needClass)
197 {
198    NamedLink64 item = null;
199    Class b;
200    int64 i64Data = 0;
201    switch(_class.typeSize)
202    {
203       case 1:
204          i64Data = !strcmp(_class.dataTypeString, "byte") ? (int64)*(byte *)data : (int64)*(char *)data;
205          break;
206       case 2:
207          i64Data = !strcmp(_class.dataTypeString, "uint16") ? (int64)*(uint16 *)data : (int64)*(short *)data;
208          break;
209       case 4:
210          i64Data = !strcmp(_class.dataTypeString, "uint") ? (int64)*(uint *)data : (int64)*(int *)data;
211          break;
212       case 8:
213          i64Data = !strcmp(_class.dataTypeString, "uint64") ? *(int64 *)data : *(int64 *)data;
214          break;
215    }
216    for(b = _class; !item && b && b.type == enumClass; b = b.base)
217    {
218       EnumClassData enumeration = (EnumClassData)b.data;
219       for(item = enumeration.values.first; item; item = item.next)
220          if(item.data == i64Data)
221             break;
222    }
223    if(item)
224    {
225       if(tempString)
226       {
227          strcpy(tempString, item.name);
228          if(!needClass || !*needClass)
229             tempString[0] = (char)toupper(tempString[0]);
230          return tempString;
231       }
232       else
233          return item.name;
234    }
235    else
236       return null;
237 }
238
239 static bool Enum_OnGetDataFromString(Class _class, void * data, const char * string)
240 {
241    NamedLink64 item = null;
242    Class b;
243    for(b = _class; !item && b && b.type == enumClass; b = b.base)
244    {
245       EnumClassData enumeration = (EnumClassData)_class.data;
246       for(item = enumeration.values.first; item; item = item.next)
247       {
248          if(item.name && !strcmpi(item.name, string))
249             break;
250       }
251    }
252    if(item)
253    {
254       switch(_class.typeSize)
255       {
256          case 1:
257             if(!strcmp(_class.dataTypeString, "byte"))
258                *(byte *)data = (byte)item.data;
259             else
260                *(char *)data = (char)item.data;
261             break;
262          case 2:
263             if(!strcmp(_class.dataTypeString, "uint16"))
264                *(uint16 *)data = (uint16)item.data;
265             else
266                *(short *)data = (short)item.data;
267             break;
268          case 4:
269             if(!strcmp(_class.dataTypeString, "uint"))
270                *(uint *)data = (uint)item.data;
271             else
272                *(int *)data = (int)item.data;
273             break;
274          case 8:
275             if(!strcmp(_class.dataTypeString, "uint64"))
276                *(uint64 *)data = *(uint64 *)&item.data;
277             else
278                *(int64 *)data = item.data;
279             break;
280       }
281       return true;
282    }
283    else
284       return Int64_OnGetDataFromString(_class, data, string);
285    return false;
286 }
287
288 static void OnFree(Class _class, void * data)
289 {
290    if(_class.templateClass) _class = _class.templateClass;
291    if(_class.type == normalClass)
292    {
293       // eInstance_Delete(data);
294       eInstance_DecRef(data);
295    }
296    else if(_class.type == noHeadClass && data)
297    {
298       while(_class && _class.type == noHeadClass)
299       {
300          if(_class.Destructor)
301             _class.Destructor(data);
302          _class = _class.base;
303       }
304       delete data;
305    }
306 }
307
308 static int DataMember_OnCompare(DataMember parentMember, void * data1, void * data2)
309 {
310    DataMember member;
311    Module module = parentMember._class.module;
312    for(member = parentMember.members.first; member; member = member.next)
313    {
314       int memberResult = 0;
315       if(member.type == normalMember)
316       {
317          Class memberType = member.dataTypeClass;
318
319          if(!memberType)
320             memberType = member.dataTypeClass = eSystem_FindClass(module, member.dataTypeString);
321          if(!memberType)
322             memberType = member.dataTypeClass = eSystem_FindClass(module, "int");
323
324          if(memberType.type == structClass || memberType.type == normalClass || memberType.type == noHeadClass)
325          {
326             memberResult = ((int (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnCompare])(memberType,
327                (byte *)data1 + member.offset,
328                (byte *)data2 + member.offset);
329             if(memberResult)
330                return memberResult;
331          }
332          else
333          {
334             DataValue value1, value2;
335             value1.i = *(int *)((byte *)data1 + member.offset);
336             value2.i = *(int *)((byte *)data2 + member.offset);
337             memberResult = ((int (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnCompare])(memberType, &value1, &value2);
338             if(memberResult)
339                return memberResult;
340          }
341       }
342       else
343       {
344          memberResult = DataMember_OnCompare(member,
345             (byte *)data1 + member.offset,
346             (byte *)data2 + member.offset);
347          if(memberResult)
348             return memberResult;
349       }
350    }
351    return 0;
352 }
353
354 static int OnCompare(Class _class, void * data1, void * data2)
355 {
356    Module module = _class.module;
357    if(_class.type == normalClass || _class.type == noHeadClass || _class.type == structClass)
358    {
359       if(data1 && data2)
360       {
361          // NOTE: Comparing from top class down here... might want to reverse it
362          for(; _class && _class.type != systemClass; _class = _class.base)
363          {
364             DataMember member;
365
366             // TESTING THIS HERE...
367             if(_class.noExpansion)
368             {
369                if(data1 > data2) return 1;
370                else if(data1 < data2) return -1;
371                else
372                   return 0;
373             }
374
375             for(member = _class.membersAndProperties.first; member; member = member.next)
376             {
377                int memberResult = 0;
378                if(member.id < 0) continue;
379                if(member.isProperty || member.type == normalMember)
380                {
381                   Class memberType = member.dataTypeClass;
382
383                   if(!memberType)
384                      memberType = member.dataTypeClass = eSystem_FindClass(module, member.dataTypeString);
385                   /*
386                   if(!memberType)
387                      memberType = member.dataTypeClass = eSystem_FindClass(module, "int");
388                   */
389                   if(memberType)
390                   {
391                      if(member.isProperty)
392                      {
393                         Property prop = (Property)member;
394                         if(!prop.conversion && prop.Get && prop.Set)
395                         {
396                            if(memberType.type == structClass || memberType.type == normalClass || memberType.type == noHeadClass)
397                            {
398                               if(!strcmp(memberType.dataTypeString, "char *"))
399                               {
400                                  String a = ((String(*)(void *))(void *)prop.Get)(data1);
401                                  String b = ((String(*)(void *))(void *)prop.Get)(data2);
402                                  memberResult = ((int (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnCompare])(memberType, a, b);
403                               }
404                            }
405                            else
406                            {
407                               DataValue value1, value2;
408                               if(!strcmp(memberType.dataTypeString, "float"))
409                               {
410                                  value1.f = ((float(*)(void *))(void *)prop.Get)(data1);
411                                  value2.f = ((float(*)(void *))(void *)prop.Get)(data2);
412                               }
413                               else
414                               {
415                                  value1.i = ((int(*)(void*))(void *)prop.Get)(data1);
416                                  value2.i = ((int(*)(void*))(void *)prop.Get)(data2);
417                               }
418                               memberResult = ((int (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnCompare])(memberType, &value1, &value2);
419                            }
420                         }
421                      }
422                      else
423                      {
424                         if(memberType.type == structClass || memberType.type == normalClass || memberType.type == noHeadClass)
425                         {
426                            if(memberType.type == normalClass || memberType.type == noHeadClass)
427                            {
428                               // TESTING THIS!
429                               memberResult = ((int (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnCompare])(memberType,
430                                  *(void **)((byte *)data1 + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset)),
431                                  *(void **)((byte *)data2 + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset)));
432                            }
433                            else
434                            {
435                               memberResult = ((int (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnCompare])(memberType,
436                                  (byte *)data1 + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset),
437                                  (byte *)data2 + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset));
438                            }
439                         }
440                         else
441                         {
442                            DataValue value1, value2;
443                            if(memberType.typeSize == 8)
444                            {
445                               value1.ui64 = *(uint64 *)((byte *)data1 + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset));
446                               value2.ui64 = *(uint64 *)((byte *)data2 + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset));
447                            }
448                            else
449                            {
450                               value1.i = *(int *)((byte *)data1 + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset));
451                               value2.i = *(int *)((byte *)data2 + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset));
452                            }
453                            memberResult = ((int (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnCompare])(memberType, &value1, &value2);
454                         }
455                      }
456                   }
457                   else
458                   {
459                      // printf("Unknown type\n");
460                   }
461                }
462                else
463                {
464                   memberResult = DataMember_OnCompare(member,
465                      (byte *)data1 + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset),
466                      (byte *)data2 + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset));
467                }
468                if(memberResult)
469                   return memberResult;
470             }
471          }
472       }
473       else if(!data1 && data2)
474          return 1;
475       else if(data1 && !data2)
476          return -1;
477    }
478    else if(_class.type == unitClass)
479    {
480       Class dataType = eSystem_FindClass(module, _class.dataTypeString);
481       return ((int (*)(void *, void *, void *))(void *)dataType._vTbl[__ecereVMethodID_class_OnCompare])(dataType, data1, data2);
482    }
483    else
484    {
485       int result = 0;
486       if(data1 && data2)
487       {
488          if(_class.typeSize == 8)
489          {
490             if(*(uint64 *)data1 > *(uint64 *)data2)
491                result = 1;
492             else if(*(uint64 *)data1 < *(uint64 *)data2)
493                result = -1;
494          }
495          else
496          {
497             if(*(uint *)data1 > *(uint *)data2)
498                result = 1;
499             else if(*(uint *)data1 < *(uint *)data2)
500                result = -1;
501          }
502       }
503       else if(!data1 && data2)
504          return 1;
505       else if(data1 && !data2)
506          return -1;
507       return result;
508    }
509    return 0;
510 }
511
512 static const char * OnGetString(Class _class, void * data, char * tempString, void * fieldData, bool * needClass)
513 {
514    // WHY DOES _class.module NOT SEEM TO WORK?
515    Module module = _class.templateClass ? _class.templateClass.module : _class.module;
516    if(_class.type == normalClass && _class.base && !_class.base.base)
517    {
518       if(sizeof(uintsize) == 8)
519          return UInt64Hex_OnGetString(_class, (void *)&data, tempString, fieldData, needClass);
520       else
521          return UIntegerHex_OnGetString(_class, (void *)&data, tempString, fieldData, needClass);
522    }
523    else if(_class.type == enumClass)
524    {
525       return Enum_OnGetString(_class, data, tempString, fieldData, needClass);
526    }
527    else if(_class.type == unitClass)
528    {
529       Class dataType;
530       Property prop;
531       for(prop = _class.conversions.first; prop; prop = prop.next)
532       {
533          bool refProp = false;
534          Class c;
535          if(!strcmp(prop.name, _class.base.fullName))
536             refProp = true;
537          else if( (c = eSystem_FindClass(_class.module, prop.name) ) )
538          {
539             Property p;
540             for(p = c.conversions.first; p; p = p.next)
541             {
542                if(!strcmp(p.name, _class.base.fullName) && !p.Set && !p.Get)
543                {
544                   refProp = true;
545                   break;
546                }
547             }
548          }
549          if(refProp)
550          {
551             if(prop.Set && prop.Get)
552             {
553                const String dts = _class.base.dataTypeString;
554                if(!strcmp(dts, "double"))
555                {
556                   double d = ((double(*)(double))(void *)prop.Set)(*(double *)data);
557                   return ((const char *(*)(void *, void *, char *, void *, bool *))(void *)class(double)._vTbl[__ecereVMethodID_class_OnGetString])(class(double), &d, tempString, fieldData, needClass);
558                }
559                else if(!strcmp(dts, "float"))
560                {
561                   float d = ((float(*)(float))(void *)prop.Set)(*(float *)data);
562                   return ((const char *(*)(void *, void *, char *, void *, bool *))(void *)class(float)._vTbl[__ecereVMethodID_class_OnGetString])(class(float), &d, tempString, fieldData, needClass);
563                }
564                else if(!strcmp(dts, "int"))
565                {
566                   int d = ((int(*)(int))(void *)prop.Set)(*(int *)data);
567                   return ((const char *(*)(void *, void *, char *, void *, bool *))(void *)class(int)._vTbl[__ecereVMethodID_class_OnGetString])(class(int), &d, tempString, fieldData, needClass);
568                }
569                else if(!strcmp(dts, "int64"))
570                {
571                   int64 d = ((int64(*)(int64))(void *)prop.Set)(*(int64 *)data);
572                   return ((const char *(*)(void *, void *, char *, void *, bool *))(void *)class(int64)._vTbl[__ecereVMethodID_class_OnGetString])(class(int64), &d, tempString, fieldData, needClass);
573                }
574             }
575             else
576                break;
577          }
578       }
579       dataType = eSystem_FindClass(module, _class.dataTypeString);
580       return ((const char *(*)(void *, void *, char *, void *, bool *))(void *)dataType._vTbl[__ecereVMethodID_class_OnGetString])(dataType, data, tempString, fieldData, needClass);
581    }
582    else
583    {
584       bool atMember = true;
585       bool prev = false;
586       Class mainClass = _class;
587       _class = null;
588       tempString[0] = '\0';
589       if(!data && (mainClass.type == normalClass || mainClass.type == noHeadClass)) return tempString;
590
591       while(_class != mainClass)
592       {
593          DataMember member;
594          Class lastClass = _class;
595
596          for(_class = mainClass; _class.base != lastClass && _class.base.type != systemClass; _class = _class.base);
597
598          for(member = _class.membersAndProperties.first; member; member = member.next)
599          {
600             char memberString[1024];
601             Class memberType = member.dataTypeClass;
602             const char * name = member.name;
603             const char *(* onGetString)(void *, void *, char *, void *, bool *);
604             if(member.id < 0) continue;
605
606             memberString[0] = 0;
607
608             if(!memberType)
609                memberType = member.dataTypeClass = eSystem_FindClass(module, member.dataTypeString);
610             if(!memberType)
611                memberType = member.dataTypeClass = eSystem_FindClass(module, "int");
612
613             onGetString = memberType._vTbl[__ecereVMethodID_class_OnGetString];
614
615             if(member.isProperty)
616             {
617                Property prop = (Property) member;
618
619                if(!prop.conversion && prop.Get && prop.Set && (!prop.IsSet || prop.IsSet(data)))
620                {
621                   if(memberType.type != structClass && (memberType.type != normalClass || !strcmp(memberType.dataTypeString, "char *")) && memberType.type != bitClass && data)
622                   {
623                      DataValue value { };
624                      if(!strcmp(prop.dataTypeString, "float"))
625                      {
626                         value.f = ((float(*)(void *))(void *)prop.Get)(data);
627                         if(value.f)
628                         {
629                            bool needClass = true;
630                            const char * result = onGetString(memberType, &value, memberString, null, &needClass);
631                            if(result && result != memberString)
632                               strcpy(memberString, result);
633                            // TESTING THIS HERE
634                            if(strchr(memberString, '.'))
635                               strcat(memberString, "f");
636                         }
637                      }
638                      else if(memberType.type == normalClass || memberType.type == noHeadClass)
639                      {
640                         value.p = ((void *(*)(void *))(void *)prop.Get)(data);
641                         if(value.p || prop.IsSet)
642                         {
643                            bool needClass = true;
644                            const char * result = onGetString(memberType,
645                               (memberType.type == normalClass) ? value.p : &value, memberString, null, &needClass);
646                            if(result && result != memberString)
647                               strcpy(memberString, result);
648                         }
649                      }
650                      else
651                      {
652                         value.i = ((int(*)(void *))(void *)prop.Get)(data);
653                         if(value.i || prop.IsSet)
654                         {
655                            bool needClass = true;
656                            const char * result = onGetString(memberType, &value, memberString, null, &needClass);
657                            if(result && result != memberString)
658                               strcpy(memberString, result);
659                         }
660                      }
661                   }
662                }
663             }
664             else
665             {
666                uint offset = member.offset + member._class.offset;
667                byte * memberData = (byte *)data + offset;
668                if(member.type == normalMember)
669                {
670                   if(memberType.type == structClass || memberType.type == normalClass)
671                   {
672                      char internalMemberString[1024];
673                      int c;
674                      uint typeSize = (memberType.type == normalClass) ? memberType.typeSize : memberType.structSize;
675                      for(c = 0; c < typeSize; c++)
676                         if(memberData[c])
677                            break;
678                      if(c < typeSize)
679                      {
680                         bool needClass = true;
681                         const char * result;
682                         if(memberType.type == normalClass)
683                            result = onGetString(memberType, *(Instance *)memberData, internalMemberString, null, &needClass);
684                         else
685                            result = onGetString(memberType, memberData, internalMemberString, null, &needClass);
686                         if(needClass && strcmp(memberType.dataTypeString, "char *"))
687                         {
688                            //strcpy(memberString, memberType.name);
689                            strcat(memberString, "{ ");
690                            if(result) strcat(memberString, result);
691                            strcat(memberString, " }");
692                         }
693                         else if(result)
694                            strcpy(memberString, result);
695                      }
696                   }
697                   //else if(_class /*memberType*/.type != bitClass)
698                   else // if(_class /*memberType*/.type != bitClass)
699                   {
700                      DataValue value { };
701                      if(_class.type == bitClass)
702                      {
703                         BitMember bitMember = (BitMember) member;
704                         // TODO: Check if base type is 32 or 64 bit
705
706                         //value.ui = (((uint)data & bitMember.mask) >> bitMember.pos);
707                         value.ui64 = ((*(uint*)data & bitMember.mask) >> bitMember.pos);
708                         if(value.ui64 && (memberType != _class))  // Avoid infinite recursion on bit classes holding themselves
709                         {
710                            bool needClass = true;
711                            char internalMemberString[1024];
712                            const char * result = onGetString(memberType, &value, internalMemberString, null, &needClass);
713
714                            if(needClass && memberType.type != systemClass && memberType.type != enumClass && memberType.type != unitClass)
715                            {
716                               //strcpy(memberString, memberType.name);
717                               strcat(memberString, " { ");
718                               if(result) strcat(memberString, result);
719                               strcat(memberString, " }");
720                            }
721                            else if(result)
722                               strcpy(memberString, result);
723                            /*
724                            if(result && memberString != result)
725                               strcpy(memberString, result);
726                            */
727                         }
728                      }
729                      else if(!memberType.noExpansion)
730                      {
731                         // TOCHECK: Is this null check still right??
732                         if(memberType.typeSize > 4 || *(int *)memberData)
733                         {
734                            bool needClass = true;
735                            const char * result = onGetString(memberType, memberData, memberString, null, &needClass);
736                            if(result && memberString != result)
737                               strcpy(memberString, result);
738                         }
739                      }
740                   }
741                   /*else
742                   {
743                      char internalMemberString[1024];
744                      byte * memberData = ((byte *)data + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset));
745                      bool needClass = true;
746                      const char * result;
747                      result = onGetString(memberType, memberData, internalMemberString, null, &needClass);
748                      if(needClass)
749                      {
750                         //strcpy(memberString, memberType.name);
751                         strcat(memberString, "{ ");
752                         if(result) strcat(memberString, result);
753                         strcat(memberString, " }");
754                      }
755                      else if(result)
756                         strcpy(memberString, result);
757                   }*/
758                }
759                // MemberUnion
760                // MemberStruct
761             }
762             // TODO: Fix atID stuff
763             if(memberString[0])
764             {
765                if(prev)
766                   strcat(tempString, ", ");
767                if(!atMember || !strcmp(memberType.name, "bool"))
768                {
769                   strcat(tempString, name);
770                   strcat(tempString, " = ");
771                }
772
773                if(!strcmp(memberType.name, "char *"))
774                {
775                   int len = strlen(tempString);
776                   int c;
777                   strcat(tempString, "\"");
778                   len ++;
779                   for(c = 0; memberString[c]; c++)
780                   {
781                      if(memberString[c] == '\"')
782                      {
783                         strcat(tempString, "\\\"");
784                         len+=2;
785                      }
786                      else if(memberString[c] == '\\')
787                      {
788                         strcat(tempString, "\\\\");
789                         len+=2;
790                      }
791                      else
792                      {
793                         tempString[len++] = memberString[c];
794                         tempString[len] = 0;
795                      }
796                   }
797                   strcat(tempString, "\"");
798                }
799                else
800                   strcat(tempString, memberString);
801                atMember = true;
802
803                prev = true;
804             }
805             else if(member && (!member.isProperty || !((Property)member).conversion))
806                atMember = false;
807          }
808       }
809    }
810    return tempString;
811 }
812
813 static bool OnGetDataFromString(Class _class, void ** data, const char * string)
814 {
815    bool result;
816    Module module = _class.module;
817    if(_class.type == enumClass)
818       result = Enum_OnGetDataFromString(_class, (int64 *)data, string);
819    else if(_class.type == unitClass)
820    {
821       Class dataType;
822       Property prop;
823       for(prop = _class.conversions.first; prop; prop = prop.next)
824       {
825          bool refProp = false;
826          Class c;
827          if(!strcmp(prop.name, _class.base.fullName))
828             refProp = true;
829          else if( (c = eSystem_FindClass(_class.module, prop.name) ) )
830          {
831             Property p;
832             for(p = c.conversions.first; p; p = p.next)
833             {
834                if(!strcmp(p.name, _class.base.fullName) && !p.Set && !p.Get)
835                {
836                   refProp = true;
837                   break;
838                }
839             }
840          }
841          if(refProp)
842          {
843             if(prop.Set && prop.Get)
844             {
845                const String dts = _class.base.dataTypeString;
846                if(!strcmp(dts, "double"))
847                {
848                   double d;
849                   bool result = ((bool (*)(void *, void *, const char *))(void *)class(double)._vTbl[__ecereVMethodID_class_OnGetDataFromString])(class(double), &d, string);
850                   *(double *)data =((double(*)(double))(void *)prop.Get)(d);
851                   return result;
852                }
853                else if(!strcmp(dts, "float"))
854                {
855                   float d;
856                   bool result = ((bool (*)(void *, void *, const char *))(void *)class(float)._vTbl[__ecereVMethodID_class_OnGetDataFromString])(class(float), &d, string);
857                   *(float *)data = ((float(*)(float))(void *)prop.Get)(d);
858                   return result;
859                }
860                else if(!strcmp(dts, "int"))
861                {
862                   int d;
863                   bool result = ((bool (*)(void *, void *, const char *))(void *)class(int)._vTbl[__ecereVMethodID_class_OnGetDataFromString])(class(int), &d, string);
864                   *(int *)data = ((int(*)(int))(void *)prop.Get)(d);
865                   return result;
866                }
867                else if(!strcmp(dts, "int64"))
868                {
869                   int64 d;
870                   bool result = ((bool (*)(void *, void *, const char *))(void *)class(int64)._vTbl[__ecereVMethodID_class_OnGetDataFromString])(class(int64), &d, string);
871                   *(int64 *)data = ((int64(*)(int64))(void *)prop.Get)(d);
872                   return result;
873                }
874             }
875             else
876                break;
877          }
878       }
879       dataType = eSystem_FindClass(module, _class.dataTypeString);
880       return ((bool (*)(void *, void *, const char *))(void *)dataType._vTbl[__ecereVMethodID_class_OnGetDataFromString])(dataType, data, string);
881    }
882    else if(!string[0] && _class.type == normalClass)
883    {
884       // result = false;  // Why was this data = null commented?
885       *data = null;
886       return true;
887    }
888    else
889    {
890       int c;
891       char memberName[1024];
892       char memberString[10240];
893       int count = 0;
894       bool quoted = false;
895       int brackets = 0;
896       char ch;
897       bool escape = false;
898       bool gotChar;
899       uint memberOffset;
900       Class curClass = null;
901       DataMember curMember = null;
902       DataMember subMemberStack[256];
903       int subMemberStackPos = 0;
904
905       result = true;
906
907       if(_class.type == noHeadClass || _class.type == normalClass)
908       {
909          data = *data = eInstance_New(_class);
910          if(_class.type == normalClass)
911             ((Instance)data)._refCount++;
912       }
913       else if(/*_class.type == noHeadClass || */_class.type == structClass)
914          memset(data, 0, _class.structSize);
915       // Bit classes cleared outside?
916
917       memberName[0] = '\0';
918
919       for(c = 0; string[c] && count < sizeof(memberString); )
920       {
921          bool found = false;
922          DataMember thisMember = null;
923
924          brackets = 0;
925          gotChar = false;
926          for(; (ch = string[c]) && count < sizeof(memberString); c++)
927          {
928             if(ch == '\"' && !escape)
929             {
930                quoted ^= true;
931             }
932             else if(quoted)
933             {
934                if(!escape && ch == '\\')
935                {
936                   escape = true;
937                }
938                else
939                {
940                   memberString[count++] = ch;
941                   escape = false;
942                }
943             }
944             // TOFIX: OnGetDataFromString is far from ready as a generic object notation reader...
945             // It has mostly been tested/used for entering data in the IDE's property sheet, as well as for parsing code in the Code Editor
946             // Is it used by the compiler?
947             else if(ch == ' ') // || ch == '\n' || ch == '\t' || ch == '\r')
948             {
949                if(gotChar)
950                   memberString[count++] = ch;
951             }
952             else if(ch == ',')
953             {
954                if(brackets)
955                {
956                   memberString[count++] = ch;
957                }
958                else
959                {
960                   c++;
961                   break;
962                }
963             }
964             else if(ch == '{')
965             {
966                // If bracket is not initialization
967                if(gotChar && !brackets)
968                {
969                   count = 0;
970                   gotChar = false;
971                }
972
973                if(brackets)
974                {
975                   memberString[count++] = ch;
976                   gotChar = true;
977                }
978                brackets++;
979             }
980             else if(ch == '}')
981             {
982                brackets--;
983                if(brackets)
984                {
985                   gotChar = true;
986                   memberString[count++] = ch;
987                }
988             }
989             else if(ch == '=')
990             {
991                if(brackets)
992                {
993                   memberString[count++] = ch;
994                }
995                else
996                {
997                   memberString[count] = '\0';
998                   //TrimLSpaces(memberString, memberName);
999                   //TrimRSpaces(memberName, memberString);
1000                   //strcpy(memberName, memberString);
1001
1002                   TrimRSpaces(memberString, memberName);
1003                   count = 0;
1004                   gotChar = false;
1005                }
1006             }
1007             else
1008             {
1009                memberString[count++] = ch;
1010                gotChar = true;
1011             }
1012          }
1013
1014          memberString[count] = '\0';
1015          TrimRSpaces(memberString, memberString);
1016
1017          if(memberName[0])
1018          {
1019             DataMember _subMemberStack[256];
1020             int _subMemberStackPos = 0;
1021
1022             thisMember = eClass_FindDataMemberAndOffset(_class, memberName, &memberOffset, _class.module, _subMemberStack, &_subMemberStackPos);
1023
1024             if(!thisMember)
1025                thisMember = (DataMember)eClass_FindProperty(_class, memberName, _class.module);
1026             if(thisMember)
1027             {
1028                if(thisMember.memberAccess == publicAccess)
1029                {
1030                   curMember = thisMember;
1031                   curClass = thisMember._class;
1032                   memcpy(subMemberStack, _subMemberStack, sizeof(int) * _subMemberStackPos);
1033                   subMemberStackPos = _subMemberStackPos;
1034                }
1035                found = true;
1036             }
1037          }
1038          else
1039          {
1040             eClass_FindNextMember(_class, &curClass, (DataMember *)&curMember, subMemberStack, &subMemberStackPos);
1041             thisMember = curMember;
1042             if(thisMember)
1043             {
1044                found = true;
1045                eClass_FindDataMemberAndOffset(_class, thisMember.name, &memberOffset, _class.module, null, null);
1046             }
1047          }
1048          if(found)
1049          {
1050             Class memberType = thisMember.dataTypeClass;
1051             uint offset;
1052             byte * memberData;
1053
1054             if(!memberType)
1055                memberType = thisMember.dataTypeClass = eSystem_FindClass(module, thisMember.dataTypeString);
1056             if(!memberType)
1057                memberType = thisMember.dataTypeClass = eSystem_FindClass(module, "int");
1058             offset = thisMember._class.offset + memberOffset;
1059             memberData = (byte *)data + offset;
1060             if(memberType.type == structClass)
1061             {
1062                if(thisMember)
1063                {
1064                   if(!((bool (*)(void *, void *, const char *))(void *)memberType._vTbl[__ecereVMethodID_class_OnGetDataFromString])(memberType, memberData, memberString))
1065                      result = false;
1066                }
1067             }
1068             else
1069             {
1070                DataValue value { };
1071                // Patch for hotKey crash ( #556 )
1072                // Key has a member KeyCode, which inherits from Key
1073                // We don't want KeyCode to use its base class OnGetDataFromString
1074                if(memberType._vTbl[__ecereVMethodID_class_OnGetDataFromString] == _class._vTbl[__ecereVMethodID_class_OnGetDataFromString])
1075                {
1076                   if(!OnGetDataFromString(memberType, (void **)&value, memberString))
1077                      result = false;
1078                }
1079                else if(!((bool (*)(void *, void *, const char *))(void *)memberType._vTbl[__ecereVMethodID_class_OnGetDataFromString])(memberType, &value, memberString))
1080                   result = false;
1081                if(thisMember && !thisMember.isProperty)
1082                {
1083                   if(_class.type == bitClass)
1084                   {
1085                      BitMember bitMember = (BitMember) thisMember;
1086                      // TODO: Check if bit _class is 32 or 64 bit
1087                      *(uint *)data = (uint32)(((*(uint *)data & ~bitMember.mask)) | ((value.ui64<<bitMember.pos)&bitMember.mask));
1088                   }
1089                   else
1090                      *(int *)memberData = value.i;
1091                }
1092                else if(thisMember.isProperty && ((Property)thisMember).Set)
1093                {
1094                   if(memberType.type == noHeadClass || memberType.type == normalClass || memberType.type == structClass)
1095                      ((void (*)(void *, void *))(void *)((Property)thisMember).Set)(data, value.p);
1096                   else
1097                   {
1098                      // TODO: Complete and improve this type of stuff throughout
1099                      if(!strcmp(memberType.dataTypeString, "float"))
1100                      {
1101                         ((void (*)(void *, float))(void *)((Property)thisMember).Set)(data, value.f);
1102                      }
1103                      else if(!strcmp(memberType.dataTypeString, "double"))
1104                      {
1105                         ((void (*)(void *, double))(void *)((Property)thisMember).Set)(data, value.d);
1106                      }
1107                      else if(!strcmp(memberType.dataTypeString, "int64"))
1108                      {
1109                         ((void (*)(void *, int64))(void *)((Property)thisMember).Set)(data, value.i64);
1110                      }
1111                      else
1112                      {
1113                         ((void (*)(void *, int))(void *)((Property)thisMember).Set)(data, value.i);
1114                      }
1115                   }
1116                }
1117             }
1118          }
1119          else
1120             result = false;
1121
1122          count = 0;
1123          memberName[0] = '\0';
1124       }
1125    }
1126    return result;
1127 }
1128
1129 static void OnCopy(Class _class, void ** data, void * newData)
1130 {
1131    if(_class.type == unitClass || _class.type == bitClass || _class.type == enumClass)
1132    {
1133       // An OnCopy is pointless for these, just copy the value
1134       /*
1135       Class dataType = eSystem_FindClass(_class.module, _class.dataTypeString);
1136       if(dataType)
1137          ((void (*)(void *, void *, void *))(void *)dataType._vTbl[__ecereVMethodID_class_OnCopy])(dataType, data, newData);
1138       */
1139       if(newData)
1140          memcpy(data, newData, _class.typeSize);
1141       else
1142          memset(data, 0, _class.typeSize);
1143    }
1144    else if(_class.type != structClass && (_class.type != systemClass || _class.byValueSystemClass))
1145    {
1146       *data = newData;
1147    }
1148    // Here we have either a structClass or a systemClass
1149    else if(newData)
1150       memcpy(data, newData, _class.typeSize);
1151    else
1152       memset(data, 0, _class.typeSize);
1153 }
1154
1155 static int DataMember_OnSerialize(DataMember parentMember, void * data, IOChannel channel)
1156 {
1157    DataMember member;
1158    Module module = parentMember._class.module;
1159    for(member = parentMember.members.first; member; member = member.next)
1160    {
1161       if(member.type == normalMember)
1162       {
1163          Class memberType = member.dataTypeClass;
1164
1165          if(!memberType)
1166             memberType = member.dataTypeClass = eSystem_FindClass(module, member.dataTypeString);
1167          if(!memberType)
1168             memberType = member.dataTypeClass = eSystem_FindClass(module, "int");
1169
1170          if(memberType.type == structClass || memberType.type == normalClass || memberType.type == noHeadClass)
1171          {
1172             ((void (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnSerialize])(memberType, (byte *)data + member.offset, channel);
1173          }
1174          else
1175          {
1176             DataValue value;
1177             value.i = *(int *)((byte *)data + member.offset);
1178             ((void (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnSerialize])(memberType, &value, channel);
1179          }
1180       }
1181       else
1182       {
1183          DataMember_OnSerialize(member, (byte *)data + member.offset, channel);
1184       }
1185    }
1186    return 0;
1187 }
1188
1189 static void OnSerialize(Class _class, void * data, IOChannel channel)
1190 {
1191    Module module = _class.module;
1192    // TO IMPROVE: Inherit from Unit class for better performance?
1193    if(_class.type == unitClass || _class.type == bitClass || _class.type == enumClass)
1194    {
1195       Class dataType = eSystem_FindClass(module, _class.dataTypeString);
1196       if(dataType)
1197          ((void (*)(void *, void *, void *))(void *)dataType._vTbl[__ecereVMethodID_class_OnSerialize])(dataType, data, channel);
1198    }
1199    else if(_class.type == normalClass || _class.type == noHeadClass || _class.type == structClass)
1200    {
1201       //if(data)
1202       {
1203          Class lastClass = null;
1204          while(lastClass != _class)
1205          {
1206             DataMember member;
1207             Class c;
1208             for(c = _class; c && (!c.base || c.base.type != systemClass) && c.base != lastClass; c = c.base);
1209             lastClass = c;
1210
1211             for(member = c.membersAndProperties.first; member; member = member.next)
1212             {
1213                if(member.id < 0) continue;
1214                if(member.isProperty || member.type == normalMember)
1215                {
1216                   Class memberType = member.dataTypeClass;
1217
1218                   if(!memberType)
1219                      memberType = member.dataTypeClass = eSystem_FindClass(module, member.dataTypeString);
1220                   if(memberType)
1221                   {
1222                      if(member.isProperty)
1223                      {
1224                         /*Property prop = (Property)member;
1225                         if(!prop.conversion && prop.Get && prop.Set)
1226                         {
1227                            if(memberType.type == structClass || memberType.type == normalClass || memberType.type == noHeadClass)
1228                            {
1229
1230                            }
1231                            else
1232                            {
1233                               DataValue value;
1234                               if(!strcmp(memberType.dataTypeString, "float"))
1235                               {
1236                                  value.f = ((float(*)())(void *)prop.Get)(data);
1237                               }
1238                               else
1239                               {
1240                                  value.i = prop.Get(data);
1241                               }
1242                               ((void (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnSerialize])(memberType, &value, channel);
1243                            }
1244                         }*/
1245                      }
1246                      else
1247                      {
1248                         if(!strcmp(memberType.name, "String") || memberType.type == normalClass || memberType.type == noHeadClass)
1249                         {
1250                            ((void (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnSerialize])(memberType, data ? (*(void **)((byte *)data + member._class.offset + member.offset)) : null, channel);
1251                         }
1252                         else
1253                            ((void (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnSerialize])(memberType, data ? (((byte *)data + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset))) : null, channel);
1254                      }
1255                   }
1256                   else
1257                   {
1258                      // printf("Unknown type\n");
1259                   }
1260                }
1261                else
1262                {
1263                   DataMember_OnSerialize(member, data ? ((byte *)data + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset)) : null, channel);
1264                }
1265             }
1266          }
1267       }
1268    }
1269 }
1270
1271 static int DataMember_OnUnserialize(DataMember parentMember, void * data, IOChannel channel)
1272 {
1273    DataMember member;
1274    Module module = parentMember._class.module;
1275    for(member = parentMember.members.first; member; member = member.next)
1276    {
1277       if(member.type == normalMember)
1278       {
1279          Class memberType = member.dataTypeClass;
1280
1281          if(!memberType)
1282             memberType = member.dataTypeClass = eSystem_FindClass(module, member.dataTypeString);
1283          if(!memberType)
1284             memberType = member.dataTypeClass = eSystem_FindClass(module, "int");
1285
1286          if(memberType.type == structClass || memberType.type == normalClass || memberType.type == noHeadClass)
1287          {
1288             ((void (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnUnserialize])(memberType, (byte *)data + member.offset, channel);
1289          }
1290          else
1291          {
1292             DataValue value;
1293             ((void (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnUnserialize])(memberType, &value, channel);
1294             *(int *)((byte *)data + member.offset) = value.i;
1295          }
1296       }
1297       else
1298       {
1299          DataMember_OnUnserialize(member, (byte *)data + member.offset, channel);
1300       }
1301    }
1302    return 0;
1303 }
1304
1305 static void OnUnserialize(Class _class, void ** data, IOChannel channel)
1306 {
1307    Module module = _class.module;
1308    if(_class.type == unitClass || _class.type == bitClass || _class.type == enumClass)
1309    {
1310       Class dataType = eSystem_FindClass(module, _class.dataTypeString);
1311       if(dataType)
1312          ((void (*)(void *, void *, void *))(void *)dataType._vTbl[__ecereVMethodID_class_OnUnserialize])(dataType, data, channel);
1313    }
1314    else if(_class.type == normalClass || _class.type == noHeadClass || _class.type == structClass)
1315    {
1316       if(data)
1317       {
1318          Class lastClass = null;
1319          if(_class.type == normalClass || _class.type == noHeadClass)
1320          {
1321             // TOFIX: Seriously!?!?? Fix me!
1322             data = *data = eInstance_New(_class);
1323             if(_class.type == normalClass)
1324                ((Instance)data)._refCount++;
1325          }
1326          else if(/*_class.type == noHeadClass || */_class.type == structClass)
1327             memset(data, 0, _class.structSize);
1328
1329          while(lastClass != _class)
1330          {
1331             DataMember member;
1332             Class c;
1333             for(c = _class; c && (!c.base || c.base.type != systemClass) && c.base != lastClass; c = c.base);
1334             lastClass = c;
1335
1336             for(member = c.membersAndProperties.first; member; member = member.next)
1337             {
1338                if(member.id < 0) continue;
1339                if(member.isProperty || member.type == normalMember)
1340                {
1341                   Class memberType = member.dataTypeClass;
1342
1343                   if(!memberType)
1344                      memberType = member.dataTypeClass = eSystem_FindClass(module, member.dataTypeString);
1345                   if(memberType)
1346                   {
1347                      if(member.isProperty)
1348                      {
1349                         /*Property prop = (Property)member;
1350                         if(!prop.conversion && prop.Get && prop.Set)
1351                         {
1352                            if(memberType.type == structClass || memberType.type == normalClass || memberType.type == noHeadClass)
1353                            {
1354
1355                            }
1356                            else
1357                            {
1358                               DataValue value;
1359                               ((void (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnUnserialize])(memberType, &value, channel);
1360                               prop.Set(data, value.i);
1361                            }
1362                         }*/
1363                      }
1364                      else
1365                         ((void (*)(void *, void *, void *))(void *)memberType._vTbl[__ecereVMethodID_class_OnUnserialize])(memberType,
1366                            (byte *)data + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset), channel);
1367                   }
1368                   else
1369                   {
1370                      // printf("Unknown type\n");
1371                   }
1372                }
1373                else
1374                {
1375                   DataMember_OnUnserialize(member, (byte *)data + (((member._class.type == normalClass) ? member._class.offset : 0) + member.offset), channel);
1376                }
1377             }
1378          }
1379       }
1380    }
1381 }
1382
1383 // Integer
1384 static int Integer_OnCompare(Class _class, int * data1, int * data2)
1385 {
1386    int result = 0;
1387    if(!data1 && !data2) result = 0;
1388    else if(data1 && !data2) result = 1;
1389    else if(!data1 && data2) result = -1;
1390    else if(*data1 > *data2) result = 1;
1391    else if(*data1 < *data2) result = -1;
1392    return result;
1393 }
1394
1395 /*static */const char * Integer_OnGetString(Class _class, int * data, char * string, void * fieldData, bool * needClass)
1396 {
1397    sprintf(string, "%d", *data);
1398    return string;
1399 }
1400
1401 static bool Integer_OnGetDataFromString(Class _class, int * data, const char * string)
1402 {
1403    char * end;
1404    int result = (int)strtol(string, &end, 0);
1405
1406    if(end > string)
1407    {
1408       *data = result;
1409       return true;
1410    }
1411    return false;
1412 }
1413
1414 static const char * Int16_OnGetString(Class _class, short * data, char * string, void * fieldData, bool * needClass)
1415 {
1416    sprintf(string, "%d", (int)*data);
1417    return string;
1418 }
1419
1420 static bool Int16_OnGetDataFromString(Class _class, short * data, const char * string)
1421 {
1422    char * end;
1423    short result = (short)strtol(string, &end, 0);
1424
1425    if(end > string)
1426    {
1427       *data = result;
1428       return true;
1429    }
1430    return false;
1431 }
1432
1433 static int Int16_OnCompare(Class _class, short * data1, short * data2)
1434 {
1435    int result = 0;
1436    if(!data1 && !data2) result = 0;
1437    else if(data1 && !data2) result = 1;
1438    else if(!data1 && data2) result = -1;
1439    else if(*data1 > *data2) result = 1;
1440    else if(*data1 < *data2) result = -1;
1441    return result;
1442 }
1443
1444 static int UInteger_OnCompare(Class _class, unsigned int * data1, unsigned int * data2)
1445 {
1446    int result = 0;
1447    if(!data1 && !data2) result = 0;
1448    else if(data1 && !data2) result = 1;
1449    else if(!data1 && data2) result = -1;
1450    else if(*data1 > *data2) result = 1;
1451    else if(*data1 < *data2) result = -1;
1452    return result;
1453 }
1454
1455 static const char * UInteger_OnGetString(Class _class, unsigned int * data, char * string, void * fieldData, bool * needClass)
1456 {
1457    sprintf(string, "%u", *data);
1458    return string;
1459 }
1460
1461 static int UInt16_OnCompare(Class _class, uint16 * data1, unsigned int * data2)
1462 {
1463    int result = 0;
1464    if(!data1 && !data2) result = 0;
1465    else if(data1 && !data2) result = 1;
1466    else if(!data1 && data2) result = -1;
1467    else if(*data1 > *data2) result = 1;
1468    else if(*data1 < *data2) result = -1;
1469    return result;
1470 }
1471
1472 static const char * UInt16_OnGetString(Class _class, uint16 * data, char * string, void * fieldData, bool * needClass)
1473 {
1474    sprintf(string, "%u", (uint)*data);
1475    return string;
1476 }
1477
1478
1479 static const char * UIntegerHex_OnGetString(Class _class, unsigned int * data, char * string, void * fieldData, bool * needClass)
1480 {
1481    sprintf(string, "%x", *data);
1482    return string;
1483 }
1484
1485 static bool UInteger_OnGetDataFromString(Class _class, unsigned int * data, const char * string)
1486 {
1487    char * end;
1488    uint result = (uint)strtoul(string, &end, 0);
1489    if(end > string)
1490    {
1491       *data = result;
1492       return true;
1493    }
1494    return false;
1495 }
1496
1497 static bool UInt16_OnGetDataFromString(Class _class, uint16 * data, const char * string)
1498 {
1499    char * end;
1500    uint16 result = (uint16)strtoul(string, &end, 0);
1501    if(end > string)
1502    {
1503       *data = result;
1504       return true;
1505    }
1506    return false;
1507 }
1508
1509 static int Byte_OnCompare(Class _class, byte * data1, byte * data2)
1510 {
1511    int result = 0;
1512    if(!data1 && !data2) result = 0;
1513    else if(data1 && !data2) result = 1;
1514    else if(!data1 && data2) result = -1;
1515    else if(*data1 > *data2) result = 1;
1516    else if(*data1 < *data2) result = -1;
1517    return result;
1518 }
1519
1520 static const char * Byte_OnGetString(Class _class, byte * data, char * string, void * fieldData, bool * needClass)
1521 {
1522    sprintf(string, "%u", (int)*data);
1523    return string;
1524 }
1525
1526 static const char * Char_OnGetString(Class _class, char * data, char * string, void * fieldData, bool * needClass)
1527 {
1528    if(needClass && *needClass)
1529    {
1530       char ch = *data;
1531       if(ch == '\t')      strcpy(string, "'\\t'");
1532       else if(ch == '\n') strcpy(string, "'\\n'");
1533       else if(ch == '\r') strcpy(string, "'\\r'");
1534       else if(ch == '\a') strcpy(string, "'\\a'");
1535       else if(ch == '\\') strcpy(string, "'\\\\'");
1536       else if(ch < 32 || ch >= 127)    sprintf(string, "'\\x%x'", ch);
1537       else sprintf(string, "'%c'", ch);
1538    }
1539    else
1540       sprintf(string, "%c", *data);
1541    return string;
1542 }
1543
1544 static bool Byte_OnGetDataFromString(Class _class, byte * data, const char * string)
1545 {
1546    char * end;
1547    byte result = (byte)strtoul(string, &end, 0);
1548    if(end > string)
1549    {
1550       *data = result;
1551       return true;
1552    }
1553    return false;
1554 }
1555
1556 static int Int64_OnCompare(Class _class, int64 * data1, int64 * data2)
1557 {
1558    int result = 0;
1559    if(!data1 && !data2) result = 0;
1560    else if(data1 && !data2) result = 1;
1561    else if(!data1 && data2) result = -1;
1562    else if(*data1 > *data2) result = 1;
1563    else if(*data1 < *data2) result = -1;
1564    return result;
1565 }
1566
1567 static int UInt64_OnCompare(Class _class, uint64 * data1, uint64 * data2)
1568 {
1569    int result = 0;
1570    if(!data1 && !data2) result = 0;
1571    else if(data1 && !data2) result = 1;
1572    else if(!data1 && data2) result = -1;
1573    else if(*data1 > *data2) result = 1;
1574    else if(*data1 < *data2) result = -1;
1575    return result;
1576 }
1577
1578 static int IntPtr64_OnCompare(Class _class, int64 data1, int64 data2)
1579 {
1580    int result = 0;
1581    if(data1 > data2) result = 1;
1582    else if(data1 < data2) result = -1;
1583    return result;
1584 }
1585
1586 static int IntPtr32_OnCompare(Class _class, int data1, int data2)
1587 {
1588    int result = 0;
1589    if(data1 > data2) result = 1;
1590    else if(data1 < data2) result = -1;
1591    return result;
1592 }
1593
1594 static int UIntPtr64_OnCompare(Class _class, uint64 data1, uint64 data2)
1595 {
1596    int result = 0;
1597    if(data1 > data2) result = 1;
1598    else if(data1 < data2) result = -1;
1599    return result;
1600 }
1601
1602 static int UIntPtr32_OnCompare(Class _class, uint32 data1, uint32 data2)
1603 {
1604    int result = 0;
1605    if(data1 > data2) result = 1;
1606    else if(data1 < data2) result = -1;
1607    return result;
1608 }
1609
1610 static const char * Int64_OnGetString(Class _class, int64 * data, char * string, void * fieldData, bool * needClass)
1611 {
1612    sprintf(string, FORMAT64D, *data);
1613    return string;
1614 }
1615
1616 static const char * UInt64_OnGetString(Class _class, uint64 * data, char * string, void * fieldData, bool * needClass)
1617 {
1618    sprintf(string, FORMAT64U, *data);
1619    return string;
1620 }
1621
1622 static const char * UInt64Hex_OnGetString(Class _class, uint64 * data, char * string, void * fieldData, bool * needClass)
1623 {
1624    sprintf(string, FORMAT64HEX, *data);
1625    return string;
1626 }
1627
1628 static const char * UIntPtr64_OnGetString(Class _class, uint64 data, char * string, void * fieldData, bool * needClass)
1629 {
1630    return UInt64Hex_OnGetString(_class, &data, string, fieldData, needClass);
1631 }
1632
1633 static const char * UIntPtr32_OnGetString(Class _class, uint data, char * string, void * fieldData, bool * needClass)
1634 {
1635    return UIntegerHex_OnGetString(_class, &data, string, fieldData, needClass);
1636 }
1637
1638 static const char * IntPtr64_OnGetString(Class _class, int64 data, char * string, void * fieldData, bool * needClass)
1639 {
1640    return Int64_OnGetString(_class, &data, string, fieldData, needClass);
1641 }
1642
1643 static const char * IntPtr32_OnGetString(Class _class, int data, char * string, void * fieldData, bool * needClass)
1644 {
1645    return Integer_OnGetString(_class, &data, string, fieldData, needClass);
1646 }
1647
1648 static bool Int64_OnGetDataFromString(Class _class, int64 * data, const char * string)
1649 {
1650    const char * end;
1651    uint64 result = _strtoi64(string, &end, 0);
1652    if(end > string)
1653    {
1654       *data = result;
1655       return true;
1656    }
1657    return false;
1658 }
1659
1660 static bool UInt64_OnGetDataFromString(Class _class, uint64 * data, const char * string)
1661 {
1662    const char * end;
1663    uint64 result = _strtoui64(string, &end, 0);
1664    if(end > string)
1665    {
1666       *data = result;
1667       return true;
1668    }
1669    return false;
1670 }
1671
1672 /*static */void Byte_OnSerialize(Class _class, byte * data, IOChannel channel)
1673 {
1674    channel.WriteData(data, 1);
1675 }
1676
1677 /*static */void Byte_OnUnserialize(Class _class, byte * data, IOChannel channel)
1678 {
1679    if(channel.ReadData(data, 1) != 1)
1680       *data = 0;
1681 }
1682
1683 /*static */void Int_OnSerialize(Class _class, int * data, IOChannel channel)
1684 {
1685    byte bytes[4];
1686    PUTXDWORD(bytes, * data);
1687    channel.WriteData(bytes, 4);
1688 }
1689
1690 /*static */void IntPtr32_OnSerialize(Class _class, int data, IOChannel channel)
1691 {
1692    byte bytes[4];
1693    PUTXDWORD(bytes, data);
1694    channel.WriteData(bytes, 4);
1695 }
1696
1697 /*static */void Int_OnUnserialize(Class _class, int * data, IOChannel channel)
1698 {
1699    byte bytes[4];
1700    if(channel.ReadData(bytes, 4) == 4)
1701       *data = GETXDWORD(bytes);
1702    else
1703       *data = 0;
1704 }
1705
1706 /*static */void Enum_OnSerialize(Class _class, int * data, IOChannel channel)
1707 {
1708    Class dataType = strcmp(_class.dataTypeString, "int") ? eSystem_FindClass(_class.module, _class.dataTypeString) : null;
1709    if(dataType)
1710       ((void (*)(void *, void *, void *))(void *)dataType._vTbl[__ecereVMethodID_class_OnSerialize])(dataType, data, channel);
1711    else
1712       Int_OnSerialize(_class, data, channel);
1713 }
1714
1715 /*static */void Enum_OnUnserialize(Class _class, int * data, IOChannel channel)
1716 {
1717    Class dataType = strcmp(_class.dataTypeString, "int") ? eSystem_FindClass(_class.module, _class.dataTypeString) : null;
1718    if(dataType)
1719       ((void (*)(void *, void *, void *))(void *)dataType._vTbl[__ecereVMethodID_class_OnUnserialize])(dataType, data, channel);
1720    else
1721       Int_OnUnserialize(_class, data, channel);
1722 }
1723
1724 /*static */void Int64_OnSerialize(Class _class, int64 * data, IOChannel channel)
1725 {
1726    byte bytes[8];
1727    PUTXQWORD(bytes, * data);
1728    channel.WriteData(bytes, 8);
1729 }
1730
1731 static void IntPtr64_OnSerialize(Class _class, int64 data, IOChannel channel)
1732 {
1733    byte bytes[8];
1734    PUTXQWORD(bytes, data);
1735    channel.WriteData(bytes, 8);
1736 }
1737
1738 /*static */void Int64_OnUnserialize(Class _class, int64 * data, IOChannel channel)
1739 {
1740    byte bytes[8];
1741    if(channel.ReadData(bytes, 8) == 8)
1742       *data = GETXQWORD(bytes);
1743    else
1744       *data = 0;
1745 }
1746
1747 /*static */void Word_OnSerialize(Class _class, uint16 * data, IOChannel channel)
1748 {
1749    byte bytes[2];
1750    PUTXWORD(bytes, * data);
1751    channel.WriteData(bytes, 2);
1752 }
1753
1754 /*static */void Word_OnUnserialize(Class _class, uint16 * data, IOChannel channel)
1755 {
1756    byte bytes[2];
1757    if(channel.ReadData(bytes, 2) == 2)
1758       *data = GETXWORD(bytes);
1759    else
1760       *data = 0;
1761 }
1762
1763 static void RegisterClass_Integer(Module module)
1764 {
1765    Class integerClass = eSystem_RegisterClass(normalClass, "int", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1766    integerClass.type = systemClass;
1767    delete (void *)integerClass.dataTypeString;
1768    integerClass.dataTypeString = CopyString("int");
1769    integerClass.structSize = 0;
1770    integerClass.typeSize = sizeof(int);
1771
1772    eClass_AddMethod(integerClass, "OnCompare", null, Integer_OnCompare, publicAccess);
1773    eClass_AddMethod(integerClass, "OnGetString", null, Integer_OnGetString, publicAccess);
1774    eClass_AddMethod(integerClass, "OnGetDataFromString", null, Integer_OnGetDataFromString, publicAccess);
1775    eClass_AddMethod(integerClass, "OnSerialize", null, Int_OnSerialize, publicAccess);
1776    eClass_AddMethod(integerClass, "OnUnserialize", null, Int_OnUnserialize, publicAccess);
1777
1778    integerClass = eSystem_RegisterClass(normalClass, "int64", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1779    integerClass.type = systemClass;
1780    integerClass.structSize = 0;
1781    integerClass.typeSize = sizeof(int64);
1782    delete (void *)integerClass.dataTypeString;
1783    integerClass.dataTypeString = CopyString("int64");
1784    eClass_AddMethod(integerClass, "OnGetString", null, Int64_OnGetString, publicAccess);
1785    eClass_AddMethod(integerClass, "OnCompare", null, Int64_OnCompare, publicAccess);
1786    eClass_AddMethod(integerClass, "OnGetDataFromString", null, Int64_OnGetDataFromString, publicAccess);
1787    eClass_AddMethod(integerClass, "OnSerialize", null, Int64_OnSerialize, publicAccess);
1788    eClass_AddMethod(integerClass, "OnUnserialize", null, Int64_OnUnserialize, publicAccess);
1789
1790    integerClass = eSystem_RegisterClass(normalClass, "uint", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1791    integerClass.type = systemClass;
1792    delete (void *)integerClass.dataTypeString;
1793    integerClass.dataTypeString = CopyString("unsigned int");
1794    integerClass.structSize = 0;
1795    integerClass.typeSize = sizeof(uint);
1796    eClass_AddMethod(integerClass, "OnCompare", null, UInteger_OnCompare, publicAccess);
1797    eClass_AddMethod(integerClass, "OnGetString", null, UInteger_OnGetString, publicAccess);
1798    eClass_AddMethod(integerClass, "OnGetDataFromString", null, UInteger_OnGetDataFromString, publicAccess);
1799    eClass_AddMethod(integerClass, "OnSerialize", null, Int_OnSerialize, publicAccess);
1800    eClass_AddMethod(integerClass, "OnUnserialize", null, Int_OnUnserialize, publicAccess);
1801
1802    integerClass = eSystem_RegisterClass(normalClass, "unsigned int", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1803    integerClass.type = systemClass;
1804    delete (void *)integerClass.dataTypeString;
1805    integerClass.dataTypeString = CopyString("unsigned int");
1806    integerClass.structSize = 0;
1807    integerClass.typeSize = sizeof(uint);
1808
1809    eClass_AddMethod(integerClass, "OnCompare", null, UInteger_OnCompare, publicAccess);
1810    eClass_AddMethod(integerClass, "OnGetString", null, UInteger_OnGetString, publicAccess);
1811    eClass_AddMethod(integerClass, "OnGetDataFromString", null, UInteger_OnGetDataFromString, publicAccess);
1812    eClass_AddMethod(integerClass, "OnSerialize", null, Int_OnSerialize, publicAccess);
1813    eClass_AddMethod(integerClass, "OnUnserialize", null, Int_OnUnserialize, publicAccess);
1814
1815    integerClass = eSystem_RegisterClass(normalClass, "uint16", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1816    integerClass.type = systemClass;
1817    delete (void *)integerClass.dataTypeString;
1818    integerClass.dataTypeString = CopyString("unsigned short");
1819    integerClass.structSize = 0;
1820    integerClass.typeSize = sizeof(uint16);
1821
1822    eClass_AddMethod(integerClass, "OnSerialize", null, Word_OnSerialize, publicAccess);
1823    eClass_AddMethod(integerClass, "OnUnserialize", null, Word_OnUnserialize, publicAccess);
1824    eClass_AddMethod(integerClass, "OnCompare", null, UInt16_OnCompare, publicAccess);
1825    eClass_AddMethod(integerClass, "OnGetString", null, UInt16_OnGetString, publicAccess);
1826    eClass_AddMethod(integerClass, "OnGetDataFromString", null, UInt16_OnGetDataFromString, publicAccess);
1827
1828    integerClass = eSystem_RegisterClass(normalClass, "short", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1829    integerClass.type = systemClass;
1830    delete (void *)integerClass.dataTypeString;
1831    integerClass.dataTypeString = CopyString("short");
1832    integerClass.structSize = 0;
1833    integerClass.typeSize = sizeof(short);
1834
1835    eClass_AddMethod(integerClass, "OnSerialize", null, Word_OnSerialize, publicAccess);
1836    eClass_AddMethod(integerClass, "OnUnserialize", null, Word_OnUnserialize, publicAccess);
1837    eClass_AddMethod(integerClass, "OnCompare", null, Int16_OnCompare, publicAccess);
1838    eClass_AddMethod(integerClass, "OnGetString", null, Int16_OnGetString, publicAccess);
1839    eClass_AddMethod(integerClass, "OnGetDataFromString", null, Int16_OnGetDataFromString, publicAccess);
1840
1841    /*
1842    integerClass = eSystem_RegisterClass(normalClass, "uint32", null, 0, 0, null, null, module, baseSystemAccess);
1843    integerClass.type = systemClass;
1844    delete (void *)integerClass.dataTypeString;
1845    integerClass.dataTypeString = CopyString("uint32");
1846    eClass_AddMethod(integerClass, "OnCompare", null, UInteger_OnCompare, publicAccess);
1847    eClass_AddMethod(integerClass, "OnGetString", null, UInteger_OnGetString, publicAccess);
1848    eClass_AddMethod(integerClass, "OnGetDataFromString", null, UInteger_OnGetDataFromString, publicAccess);
1849    */
1850    integerClass = eSystem_RegisterClass(normalClass, "uint32", "uint", 0, 0, null, null, module, baseSystemAccess, publicAccess);
1851    integerClass.type = systemClass;
1852    delete (void *)integerClass.dataTypeString;
1853    integerClass.dataTypeString = CopyString("unsigned int");
1854    integerClass.structSize = 0;
1855    integerClass.typeSize = sizeof(uint32);
1856
1857    integerClass = eSystem_RegisterClass(normalClass, "uint64", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1858    integerClass.type = systemClass;
1859    delete (void *)integerClass.dataTypeString;
1860    integerClass.dataTypeString = CopyString("uint64");
1861    integerClass.structSize = 0;
1862    integerClass.typeSize = sizeof(uint64);
1863    eClass_AddMethod(integerClass, "OnGetString", null, UInt64_OnGetString, publicAccess);
1864    eClass_AddMethod(integerClass, "OnGetDataFromString", null, UInt64_OnGetDataFromString, publicAccess);
1865    eClass_AddMethod(integerClass, "OnSerialize", null, Int64_OnSerialize, publicAccess);
1866    eClass_AddMethod(integerClass, "OnUnserialize", null, Int64_OnUnserialize, publicAccess);
1867    eClass_AddMethod(integerClass, "OnCompare", null, UInt64_OnCompare, publicAccess);
1868
1869    integerClass = eSystem_RegisterClass(normalClass, "byte", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1870    integerClass.type = systemClass;
1871    delete (void *)integerClass.dataTypeString;
1872    integerClass.dataTypeString = CopyString("unsigned char");
1873    integerClass.structSize = 0;
1874    integerClass.typeSize = sizeof(byte);
1875    eClass_AddMethod(integerClass, "OnCompare", null, Byte_OnCompare, publicAccess);
1876    eClass_AddMethod(integerClass, "OnGetString", null, Byte_OnGetString, publicAccess);
1877    eClass_AddMethod(integerClass, "OnGetDataFromString", null, Byte_OnGetDataFromString, publicAccess);
1878    eClass_AddMethod(integerClass, "OnSerialize", null, Byte_OnSerialize, publicAccess);
1879    eClass_AddMethod(integerClass, "OnUnserialize", null, Byte_OnUnserialize, publicAccess);
1880
1881    integerClass = eSystem_RegisterClass(normalClass, "char", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1882    integerClass.type = systemClass;
1883    delete (void *)integerClass.dataTypeString;
1884    integerClass.dataTypeString = CopyString("char");
1885    integerClass.structSize = 0;
1886    integerClass.typeSize = sizeof(char);
1887    eClass_AddMethod(integerClass, "OnCompare", null, Byte_OnCompare, publicAccess);
1888    eClass_AddMethod(integerClass, "OnGetString", null, Char_OnGetString, publicAccess);
1889    eClass_AddMethod(integerClass, "OnGetDataFromString", null, Byte_OnGetDataFromString, publicAccess);
1890    eClass_AddMethod(integerClass, "OnSerialize", null, Byte_OnSerialize, publicAccess);
1891    eClass_AddMethod(integerClass, "OnUnserialize", null, Byte_OnUnserialize, publicAccess);
1892
1893    integerClass = eSystem_RegisterClass(normalClass, "intsize", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1894    integerClass.type = systemClass;
1895    delete (void *)integerClass.dataTypeString;
1896    integerClass.dataTypeString = CopyString("ssize_t");
1897    integerClass.structSize = 0;
1898    integerClass.typeSize = sizeof(intsize);
1899    if(sizeof(intsize) == 8)
1900    {
1901       eClass_AddMethod(integerClass, "OnGetString", null, Int64_OnGetString, publicAccess);
1902       eClass_AddMethod(integerClass, "OnGetDataFromString", null, Int64_OnGetDataFromString, publicAccess);
1903       eClass_AddMethod(integerClass, "OnSerialize", null, Int64_OnSerialize, publicAccess);
1904       eClass_AddMethod(integerClass, "OnUnserialize", null, Int64_OnUnserialize, publicAccess);
1905       eClass_AddMethod(integerClass, "OnCompare", null, Int64_OnCompare, publicAccess);
1906    }
1907    else
1908    {
1909       eClass_AddMethod(integerClass, "OnCompare", null, Integer_OnCompare, publicAccess);
1910       eClass_AddMethod(integerClass, "OnGetString", null, Integer_OnGetString, publicAccess);
1911       eClass_AddMethod(integerClass, "OnGetDataFromString", null, Integer_OnGetDataFromString, publicAccess);
1912       eClass_AddMethod(integerClass, "OnSerialize", null, Int_OnSerialize, publicAccess);
1913       eClass_AddMethod(integerClass, "OnUnserialize", null, Int_OnUnserialize, publicAccess);
1914    }
1915
1916    integerClass = eSystem_RegisterClass(normalClass, "uintsize", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1917    integerClass.type = systemClass;
1918    delete (void *)integerClass.dataTypeString;
1919    integerClass.dataTypeString = CopyString("size_t");
1920    integerClass.structSize = 0;
1921    integerClass.typeSize = sizeof(uintsize);
1922    if(sizeof(uintsize) == 8)
1923    {
1924       eClass_AddMethod(integerClass, "OnGetString", null, UInt64Hex_OnGetString, publicAccess);
1925       eClass_AddMethod(integerClass, "OnGetDataFromString", null, UInt64_OnGetDataFromString, publicAccess);
1926       eClass_AddMethod(integerClass, "OnSerialize", null, Int64_OnSerialize, publicAccess);
1927       eClass_AddMethod(integerClass, "OnUnserialize", null, Int64_OnUnserialize, publicAccess);
1928       eClass_AddMethod(integerClass, "OnCompare", null, UInt64_OnCompare, publicAccess);
1929    }
1930    else
1931    {
1932       eClass_AddMethod(integerClass, "OnGetString", null, UIntegerHex_OnGetString, publicAccess);
1933       eClass_AddMethod(integerClass, "OnGetDataFromString", null, UInteger_OnGetDataFromString, publicAccess);
1934       eClass_AddMethod(integerClass, "OnSerialize", null, Int_OnSerialize, publicAccess);
1935       eClass_AddMethod(integerClass, "OnUnserialize", null, Int_OnUnserialize, publicAccess);
1936       eClass_AddMethod(integerClass, "OnCompare", null, UInteger_OnCompare, publicAccess);
1937    }
1938
1939    integerClass = eSystem_RegisterClass(normalClass, "uintptr", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1940    integerClass.type = systemClass;
1941    delete (void *)integerClass.dataTypeString;
1942    integerClass.dataTypeString = CopyString("uintptr_t");
1943    integerClass.structSize = 0;
1944    integerClass.typeSize = sizeof(uintptr);
1945    integerClass.byValueSystemClass = true;
1946    if(sizeof(uintptr) == 8)
1947    {
1948       eClass_AddMethod(integerClass, "OnGetString", null, UIntPtr64_OnGetString, publicAccess);
1949       eClass_AddMethod(integerClass, "OnGetDataFromString", null, UInt64_OnGetDataFromString, publicAccess);
1950       eClass_AddMethod(integerClass, "OnSerialize", null, IntPtr64_OnSerialize, publicAccess);
1951       eClass_AddMethod(integerClass, "OnUnserialize", null, Int64_OnUnserialize, publicAccess);
1952       eClass_AddMethod(integerClass, "OnCompare", null, UIntPtr64_OnCompare, publicAccess);
1953    }
1954    else
1955    {
1956       eClass_AddMethod(integerClass, "OnGetString", null, UIntPtr32_OnGetString, publicAccess);
1957       eClass_AddMethod(integerClass, "OnGetDataFromString", null, UInteger_OnGetDataFromString, publicAccess);
1958       eClass_AddMethod(integerClass, "OnSerialize", null, IntPtr32_OnSerialize, publicAccess);
1959       eClass_AddMethod(integerClass, "OnUnserialize", null, Int_OnUnserialize, publicAccess);
1960       eClass_AddMethod(integerClass, "OnCompare", null, UIntPtr32_OnCompare, publicAccess);
1961    }
1962
1963    integerClass = eSystem_RegisterClass(normalClass, "intptr", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
1964    integerClass.type = systemClass;
1965    delete (void *)integerClass.dataTypeString;
1966    integerClass.dataTypeString = CopyString("intptr_t");
1967    integerClass.structSize = 0;
1968    integerClass.typeSize = sizeof(intptr);
1969    integerClass.byValueSystemClass = true;
1970    if(sizeof(intptr) == 8)
1971    {
1972       eClass_AddMethod(integerClass, "OnGetString", null, IntPtr64_OnGetString, publicAccess);
1973       eClass_AddMethod(integerClass, "OnGetDataFromString", null, Int64_OnGetDataFromString, publicAccess);
1974       eClass_AddMethod(integerClass, "OnSerialize", null, IntPtr64_OnSerialize, publicAccess);
1975       eClass_AddMethod(integerClass, "OnUnserialize", null, Int64_OnUnserialize, publicAccess);
1976       eClass_AddMethod(integerClass, "OnCompare", null, IntPtr64_OnCompare, publicAccess);
1977    }
1978    else
1979    {
1980       eClass_AddMethod(integerClass, "OnGetString", null, IntPtr32_OnGetString, publicAccess);
1981       eClass_AddMethod(integerClass, "OnGetDataFromString", null, Integer_OnGetDataFromString, publicAccess);
1982       eClass_AddMethod(integerClass, "OnSerialize", null, IntPtr32_OnSerialize, publicAccess);
1983       eClass_AddMethod(integerClass, "OnUnserialize", null, Int_OnUnserialize, publicAccess);
1984       eClass_AddMethod(integerClass, "OnCompare", null, IntPtr32_OnCompare, publicAccess);
1985    }
1986 }
1987
1988 // Float
1989 static int Float_OnCompare(Class _class, float * data1, float * data2)
1990 {
1991    int result = 0;
1992    if(!data1 && !data2) result = 0;
1993    else if(data1 && !data2) result = 1;
1994    else if(!data1 && data2) result = -1;
1995    else if(*data1 > *data2) result = 1;
1996    else if(*data1 < *data2) result = -1;
1997    return result;
1998 }
1999
2000 static char * Float_OnGetString(Class _class, float * data, char * string, void * fieldData, bool * needClass)
2001 {
2002    float f = *data;
2003    if(f.isInf)
2004    {
2005       if(f.signBit)
2006          strcpy(string, "-inf");
2007       else
2008          strcpy(string, "inf");
2009    }
2010    else if(f.isNan)
2011    {
2012       if(f.signBit)
2013          strcpy(string, "-nan");
2014       else
2015          strcpy(string, "nan");
2016    }
2017    else
2018    {
2019       int c;
2020       int last = 0;
2021       bool checkFor1 = true, checkFor9 = true;
2022       int numDigits = 7, num = 1;
2023       int first9 = 0;
2024       char format[10];
2025       char * dot;
2026       int len;
2027       while(numDigits && num < f) numDigits--, num *= 10;
2028       sprintf(format, "%%.%df", numDigits);
2029
2030       //sprintf(string, "%f", f);
2031       sprintf(string, format, f);
2032       dot = strchr(string, '.');
2033
2034       len = strlen(string);
2035       c = len-1;
2036       for( ; c >= 0; c--)
2037       {
2038          char ch = string[c];
2039          if(ch != '0' && dot)
2040          {
2041             if(ch == '1' && string + c - dot >= 6 && c == len - 1 && checkFor1)
2042                checkFor1 = false;
2043             else if(ch == '9' && string + c - dot >= 6 && c == len - 1 && checkFor9)
2044                first9 = c;
2045             else
2046             {
2047                last = Max(last, c);
2048                checkFor9 = false;
2049                checkFor1 = false;
2050             }
2051          }
2052          if(ch == '.')
2053          {
2054             if(last == c)
2055                string[c] = 0;
2056             else
2057             {
2058                string[last+1] = 0;
2059                if(first9)
2060                {
2061                   while(--first9 > 0)
2062                   {
2063                      if(first9 != c)
2064                         if(string[first9] < '9')
2065                         {
2066                            string[first9]++;
2067                            break;
2068                         }
2069                   }
2070                   if(first9 < c)
2071                   {
2072                      string[c-1] = '1';
2073                      first9 = c;
2074                   }
2075                   string[first9] = 0;
2076                }
2077             }
2078             break;
2079          }
2080       }
2081    }
2082    return string;
2083 }
2084
2085 static bool Float_OnGetDataFromString(Class _class, float * data, char * string)
2086 {
2087    char * end;
2088    float result = (float)strtod(string, &end);
2089    //*data = atof(string);
2090    if(end > string)
2091    {
2092       *data = result;
2093       return true;
2094    }
2095    return false;
2096 }
2097
2098 static void Float_OnSerialize(Class _class, float * data, IOChannel channel)
2099 {
2100    byte bytes[4];
2101    PUTXDWORD(bytes, * (uint *)data);
2102    channel.WriteData(bytes, 4);
2103 }
2104
2105 static void Float_OnUnserialize(Class _class, float * data, IOChannel channel)
2106 {
2107    byte bytes[4];
2108    if(channel.ReadData(bytes, 4) == 4)
2109       *(uint *)data = GETXDWORD(bytes);
2110    else
2111       *data = 0;
2112 }
2113
2114 static void RegisterClass_Float(Module module)
2115 {
2116    Class floatClass = eSystem_RegisterClass(normalClass, "float", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
2117    floatClass.type = systemClass;
2118    delete (void *)floatClass.dataTypeString;
2119    floatClass.dataTypeString = CopyString("float");
2120    floatClass.structSize = 0;
2121    floatClass.typeSize = sizeof(float);
2122    eClass_AddMethod(floatClass, "OnCompare", null, Float_OnCompare, publicAccess);
2123    eClass_AddMethod(floatClass, "OnGetString", null, Float_OnGetString, publicAccess);
2124    eClass_AddMethod(floatClass, "OnGetDataFromString", null, Float_OnGetDataFromString, publicAccess);
2125    eClass_AddMethod(floatClass, "OnSerialize", null, Float_OnSerialize, publicAccess);
2126    eClass_AddMethod(floatClass, "OnUnserialize", null, Float_OnUnserialize, publicAccess);
2127
2128    eClass_AddMethod(floatClass, "nan", "float ::nan(void)", Float_nan, publicAccess);
2129    eClass_AddMethod(floatClass, "inf", "float ::inf(void)", Float_inf, publicAccess);
2130
2131    eClass_AddProperty(floatClass, "isNan", "bool", null, Float_isNan, publicAccess);
2132    eClass_AddProperty(floatClass, "isInf", "bool", null, Float_isInf, publicAccess);
2133    eClass_AddProperty(floatClass, "signBit", "int", null, Float_signBit, publicAccess);
2134 }
2135
2136 // Double
2137 static int Double_OnCompare(Class _class, double * data1, double * data2)
2138 {
2139    int result = 0;
2140    if(!data1 && !data2) result = 0;
2141    else if(data1 && !data2) result = 1;
2142    else if(!data1 && data2) result = -1;
2143    else if(*data1 > *data2) result = 1;
2144    else if(*data1 < *data2) result = -1;
2145    return result;
2146 }
2147
2148 static char * Double_OnGetString(Class _class, double * data, char * string, void * fieldData, bool * needClass)
2149 {
2150    double f = *data;
2151    if(f.isInf)
2152    {
2153       if(f.signBit)
2154          strcpy(string, "-inf");
2155       else
2156          strcpy(string, "inf");
2157    }
2158    else if(f.isNan)
2159    {
2160       if(f.signBit)
2161          strcpy(string, "-nan");
2162       else
2163          strcpy(string, "nan");
2164    }
2165    else
2166    {
2167       int c;
2168       int last = 0;
2169       //sprintf(string, "%.20f", f);
2170       if(runtimePlatform == win32)
2171       // sprintf(string, "%.16g", f);
2172          sprintf(string, "%.15g", f);
2173       else
2174          sprintf(string, "%.13lf", f);
2175
2176       c = strlen(string)-1;
2177       for( ; c >= 0; c--)
2178       {
2179          if(string[c] != '0')
2180             last = Max(last, c);
2181          if(string[c] == '.')
2182          {
2183             if(last == c)
2184                string[c] = 0;
2185             else
2186                string[last+1] = 0;
2187             break;
2188          }
2189       }
2190    }
2191    return string;
2192 }
2193
2194 static bool Double_OnGetDataFromString(Class _class, double * data, char * string)
2195 {
2196    char * end;
2197    double result;
2198    //*data = (double)strtod(string, null);
2199    result = strtod(string, &end);
2200    if(end > string)
2201    {
2202       *data = result;
2203       return true;
2204
2205     }
2206     return false;
2207 }
2208
2209 static void Double_OnSerialize(Class _class, double * data, IOChannel channel)
2210 {
2211    byte bytes[8];
2212    PUTXQWORD(bytes, * (uint64 *)data);
2213    channel.WriteData(bytes, 8);
2214 }
2215
2216 static void Double_OnUnserialize(Class _class, double * data, IOChannel channel)
2217 {
2218    byte bytes[8];
2219    if(channel.ReadData(bytes, 8) == 8)
2220       *(uint64 *)data = GETXQWORD(bytes);
2221    else
2222       *data = 0;
2223 }
2224
2225 static void RegisterClass_Double(Module module)
2226 {
2227    Class doubleClass = eSystem_RegisterClass(normalClass, "double", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
2228    doubleClass.type = systemClass;
2229    delete (void *)doubleClass.dataTypeString;
2230    doubleClass.dataTypeString = CopyString("double");
2231    doubleClass.structSize = 0;
2232    doubleClass.typeSize = sizeof(double);
2233    eClass_AddMethod(doubleClass, "OnCompare", null, Double_OnCompare, publicAccess);
2234    eClass_AddMethod(doubleClass, "OnGetString", null, Double_OnGetString, publicAccess);
2235    eClass_AddMethod(doubleClass, "OnGetDataFromString", null, Double_OnGetDataFromString, publicAccess);
2236    eClass_AddMethod(doubleClass, "OnSerialize", null, Double_OnSerialize, publicAccess);
2237    eClass_AddMethod(doubleClass, "OnUnserialize", null, Double_OnUnserialize, publicAccess);
2238
2239    eClass_AddProperty(doubleClass, "isNan", "bool", null, Double_isNan, publicAccess);
2240    eClass_AddProperty(doubleClass, "isInf", "bool", null, Double_isInf, publicAccess);
2241    eClass_AddProperty(doubleClass, "signBit", "int", null, Double_signBit, publicAccess);
2242
2243    eClass_AddMethod(doubleClass, "nan", "double ::nan(void)", Double_nan, publicAccess);
2244    eClass_AddMethod(doubleClass, "inf", "double ::inf(void)", Double_inf, publicAccess);
2245 }
2246
2247 public struct StaticString
2248 {
2249    char string[1];
2250
2251    void OnSerialize(IOChannel channel)
2252    {
2253       uint len = this ? strlen(string) : 0;
2254       channel.WriteData(this ? string : "", len+1);
2255    }
2256
2257    void OnUnserialize(IOChannel channel)
2258    {
2259       if(this)
2260       {
2261          int c;
2262
2263          for(c = 0; channel.ReadData(&string[c], 1) && string[c]; c++);
2264          string[c++] = '\0';
2265       }
2266    }
2267
2268    int OnCompare(StaticString string2)
2269    {
2270       int result = 0;
2271       if(this && string2)
2272          result = strcmpi(string, string2.string);
2273       else if(!this && string2)
2274          result = -1;   // INVERTED THESE 2 CASES
2275       else if(this && !string2)
2276          result = 1;
2277       return result;
2278    }
2279
2280    const char * OnGetString(char * tempString, void * fieldData, bool * needClass)
2281    {
2282       return this ? string : null;
2283    }
2284
2285    void OnFree()
2286    {
2287    }
2288 };
2289
2290 static void String_OnCopy(Class _class, char ** data, char * newData)
2291 {
2292    if(newData)
2293    {
2294       int len = strlen(newData);
2295       if(len)
2296       {
2297          *data = eSystem_New(len+1);
2298          memcpy(*data, newData, len+1);
2299       }
2300       else
2301          *data = null;
2302    }
2303    else
2304       *data = null;
2305 }
2306
2307 static bool String_OnGetDataFromString(Class _class, char ** data, char * newData)
2308 {
2309    if(newData)
2310    {
2311       int len = strlen(newData);
2312       if(len)
2313       {
2314          *data = eSystem_New(len+1);
2315          memcpy(*data, newData, len+1);
2316       }
2317       else
2318          *data = null;
2319    }
2320    return true;
2321 }
2322
2323 /*static */int String_OnCompare(Class _class, const char * string1, const char * string2)
2324 {
2325    int result = 0;
2326    if(string1 && string2)
2327       result = strcmpi(string1, string2);
2328    else if(!string1 && string2)
2329       result = 1;
2330    else if(string1 && !string2)
2331       result = -1;
2332    return result;
2333 }
2334
2335 static char * String_OnGetString(Class _class, char * string, char * tempString, void * fieldData, bool * needClass)
2336 {
2337    return string;
2338 }
2339
2340 static void String_OnFree(Class _class, char * string)
2341 {
2342    if(string)
2343    {
2344       eSystem_Delete(string);
2345    }
2346 }
2347
2348 static void String_OnSerialize(Class _class, char * string, IOChannel channel)
2349 {
2350    int len = string ? strlen(string) : 0;
2351    channel.WriteData(string ? string : "", len+1);
2352 }
2353
2354 static void String_OnUnserialize(Class _class, char * * string, IOChannel channel)
2355 {
2356    if(string)
2357    {
2358       int c;
2359       uint size = 64;
2360
2361       *string = new char[size];
2362       for(c = 0; channel.ReadData(&(*string)[c], 1) && (*string)[c]; c++)
2363       {
2364          if(c == size - 1)
2365          {
2366             size += size / 2;
2367             *string = renew *string char[size];
2368          }
2369       }
2370       (*string)[c++] = '\0';
2371       if(!UTF8Validate(*string))
2372       {
2373          char * newString = new char[c*2];
2374          ISO8859_1toUTF8(*string, newString, c*2);
2375          delete *string;
2376          *string = renew newString char[strlen(newString)+1];
2377       }
2378       else
2379          *string = renew *string char[c];
2380    }
2381 }
2382
2383 static void RegisterClass_String(Module module)
2384 {
2385    Class stringClass = eSystem_RegisterClass(normalClass, "char *", null, 0, 0, null, null, module, baseSystemAccess, publicAccess);
2386    delete (void *)stringClass.dataTypeString;
2387    stringClass.dataTypeString = CopyString("char *");
2388    stringClass.structSize = 0;
2389    stringClass.computeSize = false;
2390
2391    eClass_AddMethod(stringClass, "OnCompare", null, String_OnCompare, publicAccess);
2392    eClass_AddMethod(stringClass, "OnCopy", null, String_OnCopy, publicAccess);
2393    eClass_AddMethod(stringClass, "OnFree", null, String_OnFree, publicAccess);
2394    eClass_AddMethod(stringClass, "OnGetString", null, String_OnGetString, publicAccess);
2395    eClass_AddMethod(stringClass, "OnGetDataFromString", null, String_OnGetDataFromString, publicAccess);
2396    eClass_AddMethod(stringClass, "OnSerialize", null, String_OnSerialize, publicAccess);
2397    eClass_AddMethod(stringClass, "OnUnserialize", null, String_OnUnserialize, publicAccess);
2398    // eClass_AddProperty(stringClass, null, "Class", null, String_GetClass, publicAccess);
2399
2400    stringClass = eSystem_RegisterClass(normalClass, "String", "char *", 0, 0, null, null, module, baseSystemAccess, publicAccess);
2401    stringClass.structSize = 0;
2402    stringClass.computeSize = false;
2403    eClass_AddProperty(stringClass, null, "char *", null, null, publicAccess);
2404 }
2405
2406 void InitializeDataTypes1(Module module)
2407 {
2408    Class baseClass = eSystem_FindClass(module, "class");
2409
2410    eClass_AddVirtualMethod(baseClass, "OnDisplay", "void typed_object::OnDisplay(Surface surface, int x, int y, int width, void * fieldData, Alignment alignment, DataDisplayFlags displayFlags)", null, publicAccess);
2411    eClass_AddVirtualMethod(baseClass, "OnCompare", "int typed_object::OnCompare(any_object object)", OnCompare, publicAccess);
2412    eClass_AddVirtualMethod(baseClass, "OnCopy", "void typed_object&::OnCopy(any_object newData)", OnCopy, publicAccess);
2413    eClass_AddVirtualMethod(baseClass, "OnFree", "void typed_object::OnFree(void)", OnFree, publicAccess);
2414    eClass_AddVirtualMethod(baseClass, "OnGetString", "const char * typed_object::OnGetString(char * tempString, void * fieldData, bool * needClass)", OnGetString, publicAccess);
2415    eClass_AddVirtualMethod(baseClass, "OnGetDataFromString", "bool typed_object&::OnGetDataFromString(const char * string)", OnGetDataFromString, publicAccess);
2416    eClass_AddVirtualMethod(baseClass, "OnEdit", "Window typed_object::OnEdit(DataBox dataBox, DataBox obsolete, int x, int y, int w, int h, void * userData)", null, publicAccess);
2417    eClass_AddVirtualMethod(baseClass, "OnSerialize", "void typed_object::OnSerialize(IOChannel channel)", OnSerialize, publicAccess);
2418    eClass_AddVirtualMethod(baseClass, "OnUnserialize", "void typed_object&::OnUnserialize(IOChannel channel)", OnUnserialize, publicAccess);
2419    eClass_AddVirtualMethod(baseClass, "OnSaveEdit", "bool typed_object&::OnSaveEdit(Window window, void * object)", null, publicAccess);
2420 }
2421
2422 void InitializeDataTypes(Module module)
2423 {
2424    Class enumClass = eSystem_FindClass(module, "enum");
2425
2426    eClass_AddMethod(enumClass, "OnSerialize", null, Enum_OnSerialize, publicAccess);
2427    eClass_AddMethod(enumClass, "OnUnserialize", null, Enum_OnUnserialize, publicAccess);
2428
2429    // Data Types
2430    RegisterClass_Integer(module);
2431    RegisterClass_Float(module);
2432    RegisterClass_Double(module);
2433    RegisterClass_String(module);
2434 }
2435
2436 public int PrintStdArgsToBuffer(char * buffer, int maxLen, typed_object object, va_list args)
2437 {
2438    int len = 0;
2439    // TOFIX: OnGetString will need a maxLen as well
2440    const char * result = object.OnGetString(buffer, null, null);
2441    if(result)
2442    {
2443       len = strlen(result);
2444       if(len >= maxLen) len = maxLen-1;
2445       if(result != buffer)
2446          memcpy(buffer, result, len);
2447    }
2448    while(true)
2449    {
2450       Class _class = null;
2451       void * data = null;
2452
2453       _class = va_arg(args, void *);
2454       if(!_class) break;
2455       data = va_arg(args, void *);
2456       if(data)
2457       {
2458          // TOFIX: OnGetString will need a maxLen as well
2459          result = ((const char *(*)(void *, void *, char *, void *, bool *))(void *)_class._vTbl[__ecereVMethodID_class_OnGetString])(_class, data, buffer + len, null, null);
2460          if(result)
2461          {
2462             int newLen = strlen(result);
2463             if(len + newLen >= maxLen) newLen = maxLen-1-len;
2464             if(result != buffer + len)
2465                memcpy(buffer + len, result, newLen);
2466             len += newLen;
2467          }
2468       }
2469    }
2470    buffer[len] = 0;
2471    return len;
2472 }
2473
2474 public int PrintBuf(char * buffer, int maxLen, typed_object object, ...)
2475 {
2476    va_list args;
2477    int len;
2478    va_start(args, object);
2479    len = PrintStdArgsToBuffer(buffer, maxLen, object, args);
2480    va_end(args);
2481    return len;
2482 }
2483
2484 public int PrintLnBuf(char * buffer, int maxLen, typed_object object, ...)
2485 {
2486    va_list args;
2487    int len;
2488    va_start(args, object);
2489    len = PrintStdArgsToBuffer(buffer, maxLen-1, object, args);
2490    buffer[len++] = '\n';
2491    buffer[len] = '\0';
2492    va_end(args);
2493    return len;
2494 }
2495
2496 public char * PrintString(typed_object object, ...)
2497 {
2498    char buffer[4096];
2499    va_list args;
2500    char * string;
2501    int len;
2502    va_start(args, object);
2503    len = PrintStdArgsToBuffer(buffer, sizeof(buffer), object, args);
2504    string = new char[len + 1];
2505    memcpy(string, buffer, len + 1);
2506    va_end(args);
2507    return string;
2508 }
2509
2510 public char * PrintLnString(typed_object object, ...)
2511 {
2512    char buffer[4096];
2513    va_list args;
2514    char * string;
2515    int len;
2516    va_start(args, object);
2517    len = PrintStdArgsToBuffer(buffer, sizeof(buffer), object, args);
2518    string = new char[len + 2];
2519    memcpy(string, buffer, len);
2520    string[len++] = '\n';
2521    string[len] = '\0';
2522    va_end(args);
2523    return string;
2524 }
2525
2526 #if defined(__ANDROID__)
2527 #include <android/log.h>
2528 #define LOGI(...) ((void)__android_log_print(ANDROID_LOG_INFO, "ecere-app", __VA_ARGS__))
2529 #endif
2530
2531 public void PrintLn(typed_object object, ...)
2532 {
2533    va_list args;
2534    char buffer[4096];
2535    va_start(args, object);
2536    PrintStdArgsToBuffer(buffer, sizeof(buffer), object, args);
2537    va_end(args);
2538 #if defined(__ANDROID__) && !defined(ECERE_NOFILE)
2539    LOGI("%s", buffer);
2540 #else
2541    puts(buffer);
2542 #endif
2543 }
2544
2545 public void Print(typed_object object, ...)
2546 {
2547    va_list args;
2548    char buffer[4096];
2549    va_start(args, object);
2550    PrintStdArgsToBuffer(buffer, sizeof(buffer), object, args);
2551    va_end(args);
2552    fputs(buffer, eC_stdout());
2553 }