autoc 1.4 → 2.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (110) hide show
  1. checksums.yaml +5 -5
  2. data/CHANGES.md +3 -0
  3. data/README.md +149 -0
  4. data/cmake/AutoC.cmake +39 -0
  5. data/lib/autoc/allocators.rb +51 -0
  6. data/lib/autoc/association.rb +126 -0
  7. data/lib/autoc/box.rb +311 -0
  8. data/lib/autoc/cmake.rb +54 -0
  9. data/lib/autoc/collection.rb +83 -110
  10. data/lib/autoc/composite.rb +333 -0
  11. data/lib/autoc/cstring.rb +263 -0
  12. data/lib/autoc/function.rb +247 -0
  13. data/lib/autoc/hash_map.rb +328 -0
  14. data/lib/autoc/hash_set.rb +339 -0
  15. data/lib/autoc/hashers.rb +102 -0
  16. data/lib/autoc/list.rb +444 -0
  17. data/lib/autoc/module.rb +434 -0
  18. data/lib/autoc/openmp.rb +15 -0
  19. data/lib/autoc/primitive.rb +27 -0
  20. data/lib/autoc/ranges.rb +707 -0
  21. data/lib/autoc/record.rb +247 -0
  22. data/lib/autoc/scaffold/docs.rb +117 -0
  23. data/lib/autoc/scaffold/generic_value.rb +86 -0
  24. data/lib/autoc/scaffold/project.rb +75 -0
  25. data/lib/autoc/scaffold/test_cstring.rb +113 -0
  26. data/lib/autoc/scaffold/test_cstring_hash_set.rb +35 -0
  27. data/lib/autoc/scaffold/test_int_box.rb +22 -0
  28. data/lib/autoc/scaffold/test_int_hash_set.rb +448 -0
  29. data/lib/autoc/scaffold/test_int_list.rb +106 -0
  30. data/lib/autoc/scaffold/test_int_vector.rb +83 -0
  31. data/lib/autoc/scaffold/test_v2v_hash_map.rb +83 -0
  32. data/lib/autoc/scaffold/test_value_hash_set.rb +60 -0
  33. data/lib/autoc/scaffold/test_value_vector.rb +146 -0
  34. data/{test/test.rb → lib/autoc/scaffold/tests.rb} +179 -158
  35. data/lib/autoc/scaffold.rb +12 -0
  36. data/lib/autoc/sequential.rb +99 -0
  37. data/lib/autoc/set.rb +331 -0
  38. data/lib/autoc/std.rb +149 -0
  39. data/lib/autoc/type.rb +93 -531
  40. data/lib/autoc/vector.rb +290 -0
  41. data/lib/autoc.rb +4 -35
  42. metadata +55 -85
  43. data/.yardopts +0 -4
  44. data/CHANGES +0 -23
  45. data/README +0 -28
  46. data/doc/AutoC/Code.html +0 -523
  47. data/doc/AutoC/Collection.html +0 -1214
  48. data/doc/AutoC/HashMap.html +0 -1441
  49. data/doc/AutoC/HashSet.html +0 -916
  50. data/doc/AutoC/Iterators/Bidirectional.html +0 -204
  51. data/doc/AutoC/Iterators/Unidirectional.html +0 -200
  52. data/doc/AutoC/Iterators.html +0 -126
  53. data/doc/AutoC/List.html +0 -1039
  54. data/doc/AutoC/Maps.html +0 -290
  55. data/doc/AutoC/Module/File.html +0 -415
  56. data/doc/AutoC/Module/Header.html +0 -437
  57. data/doc/AutoC/Module/Source.html +0 -707
  58. data/doc/AutoC/Module.html +0 -948
  59. data/doc/AutoC/Priority.html +0 -138
  60. data/doc/AutoC/Queue.html +0 -1172
  61. data/doc/AutoC/Reference.html +0 -735
  62. data/doc/AutoC/Sets.html +0 -520
  63. data/doc/AutoC/String.html +0 -1394
  64. data/doc/AutoC/TreeMap.html +0 -1565
  65. data/doc/AutoC/TreeSet.html +0 -1447
  66. data/doc/AutoC/Type.html +0 -2148
  67. data/doc/AutoC/UserDefinedType.html +0 -1047
  68. data/doc/AutoC/Vector.html +0 -987
  69. data/doc/AutoC.html +0 -331
  70. data/doc/_index.html +0 -388
  71. data/doc/class_list.html +0 -51
  72. data/doc/css/common.css +0 -1
  73. data/doc/css/full_list.css +0 -58
  74. data/doc/css/style.css +0 -481
  75. data/doc/file.CHANGES.html +0 -117
  76. data/doc/file.README.html +0 -116
  77. data/doc/file_list.html +0 -61
  78. data/doc/frames.html +0 -17
  79. data/doc/index.html +0 -116
  80. data/doc/js/app.js +0 -243
  81. data/doc/js/full_list.js +0 -216
  82. data/doc/js/jquery.js +0 -4
  83. data/doc/method_list.html +0 -1307
  84. data/doc/top-level-namespace.html +0 -112
  85. data/lib/autoc/code.rb +0 -237
  86. data/lib/autoc/collection/hash_map.rb +0 -385
  87. data/lib/autoc/collection/hash_set.rb +0 -337
  88. data/lib/autoc/collection/iterator.rb +0 -39
  89. data/lib/autoc/collection/list.rb +0 -429
  90. data/lib/autoc/collection/map.rb +0 -41
  91. data/lib/autoc/collection/queue.rb +0 -517
  92. data/lib/autoc/collection/set.rb +0 -134
  93. data/lib/autoc/collection/tree_map.rb +0 -464
  94. data/lib/autoc/collection/tree_set.rb +0 -611
  95. data/lib/autoc/collection/vector.rb +0 -336
  96. data/lib/autoc/string.rb +0 -492
  97. data/test/test_auto.c +0 -7141
  98. data/test/test_auto.h +0 -753
  99. data/test/test_char_string.rb +0 -270
  100. data/test/test_int_list.rb +0 -35
  101. data/test/test_int_tree_set.rb +0 -111
  102. data/test/test_int_vector.rb +0 -34
  103. data/test/test_value_hash_map.rb +0 -162
  104. data/test/test_value_hash_set.rb +0 -173
  105. data/test/test_value_list.rb +0 -193
  106. data/test/test_value_queue.rb +0 -275
  107. data/test/test_value_tree_map.rb +0 -176
  108. data/test/test_value_tree_set.rb +0 -173
  109. data/test/test_value_vector.rb +0 -155
  110. data/test/value.rb +0 -80
data/lib/autoc/string.rb DELETED
@@ -1,492 +0,0 @@
1
- require "autoc/code"
2
- require "autoc/type"
3
-
4
-
5
- module AutoC
6
-
7
-
8
- =begin
9
-
10
- String is wrapper around the standard null-terminated C string which has the capabilities of both a plain string and a string builder optimized for appending and incremental building.
11
-
12
- It is ought to be on par with the raw C strings performance-wise (or even exceed it since the string size is tracked).
13
-
14
- Unlike the plain C string, this String type has value type semantics but it can be turned into the reference type with {AutoC::Reference}.
15
-
16
- The String's default character type, *_CharType_*, is *_char_* although this can be changed.
17
-
18
- String generally obeys the Vector interface with respect to working with its characters and resembles the List interface when using its building capabilities.
19
-
20
- == Generated C interface
21
-
22
- === String management
23
-
24
- [cols="2*"]
25
- |===
26
- |*_void_* ~type~Copy(*_Type_* * +dst+, *_Type_* * +src+)
27
- |
28
- Create a new string +dst+ filled with a _copy_ the contents of +src+.
29
-
30
- NOTE: Previous contents of +dst+ is overwritten.
31
-
32
- |*_void_* ~type~CopyRange(*_Type_* * +dst+, *_Type_* * +src+, *_size_t_* first, *_size_t_* last)
33
- |
34
- Create a new string +dst+ filled with a part the contents of +src+ lying in the range +[first, last]+, that is including the character at position +first+ and including the character at position +last+.
35
-
36
- NOTE: Previous contents of +dst+ is overwritten.
37
-
38
- WARNING: +first+ *must not* exceed +last+ (that is, +first+ <= +last+) and both indices *must* be valid otherwise behavior is undefined. See ~type~Within().
39
-
40
- |*_void_* ~type~Ctor(*_Type_* * +self+, *_const CharType *_* +chars+)
41
- |
42
- Create a new string +self+ with a _copy_ of the null-terminated C string +chars+.
43
-
44
- NULL value of +chars+ is permitted; this case corresponds to an empty string "".
45
-
46
- NOTE: Previous contents of +self+ is overwritten.
47
-
48
- |*_void_* ~type~Dtor(*_Type_* * +self+)
49
- |
50
- Destroy string +self+.
51
-
52
- |*_int_* ~type~Equal(*_Type_* * +lt+, *_Type_* * +rt+)
53
- |
54
- Return non-zero value if strings +lt+ and +rt+ are considered equal by contents and zero value otherwise.
55
-
56
- |*_size_t_* ~type~Identify(*_Type_* * +self+)
57
- |
58
- Return hash code for string +self+.
59
- |===
60
-
61
- === Basic operations
62
-
63
- [cols=2*]
64
- |===
65
- |*_const CharType *_* ~type~Chars(*_Type_* * +self+)
66
- |
67
- Return a _read-only view_ of the string in a form of the standard C null-terminated string.
68
-
69
- NOTE: the returned value need not to be freed.
70
-
71
- WARNING: the returned value should be considered *volatile* and thus may be altered or invalidated by a subsequent call to any String method!
72
-
73
- |*_int_* ~type~Empty(*_Type_* * +self+)
74
- |
75
- Return non-zero value if string +self+ has zero length and zero value otherwise.
76
-
77
- |*_CharType_* ~type~Get(*_Type_* * +self+, *_size_t_* +index+)
78
- |
79
- Return a _copy_ of the character stored in +self+ at position +index+.
80
-
81
- WARNING: +index+ *must* be a valid index otherwise the behavior is undefined. See ~type~Within().
82
-
83
- |*_void_* ~type~Set(*_Type_* * +self+, *_size_t_* +index+, *_CharType_* +value+)
84
- |
85
-
86
- Store a _copy_ of the character +value+ in string +self+ at position +index+.
87
-
88
- WARNING: +index+ *must* be a valid index otherwise the behavior is undefined. See ~type~Within().
89
-
90
- |*_size_t_* ~type~Size(*_Type_* * +self+)
91
- |
92
- Return number of characters stored in string +self+.
93
-
94
- Note that this does not include the null terminator.
95
-
96
- |*_int_* ~type~Within(*_Type_* * +self+, *_size_t_* +index+)
97
- |
98
- Return non-zero value if +index+ is a valid character index and zero value otherwise.
99
- A valid index lies between 0 and ~type~Size()-1 inclusively.
100
- |===
101
-
102
- === String buffer operations
103
-
104
- Functions which provide the string buffer functionality.
105
- This allows the incremental building of strings without excessive storage copying/reallocation.
106
-
107
- [cols=2*]
108
- |===
109
- |*_void_* ~type~PushChars(*_Type_* * +self+, *_const CharType*_* +chars+)
110
- |
111
- Append a _copy_ of the null-terminated C string +chars+ to string +self+.
112
-
113
- |*_int_* ~type~PushChar(*_Type_* * +self+, *_CharType_* +value+)
114
- |
115
- Append a _copy_ of the character +value+ to string +self+.
116
-
117
- Return non-zero value on success and zero value on conversion error.
118
-
119
- NOTE: this convenience function applies generic formatting rules and is currently implemented as a macro; for more precise control over the formatting process use the ~type~PushFormat() function.
120
-
121
- |*_int_* ~type~PushInt(*_Type_* * +self+, *_int_* +value+)
122
- |
123
- Append string representation of the integer +value+ to string +self+.
124
-
125
- Return non-zero value on success and zero value on conversion error.
126
-
127
- NOTE: this convenience function applies generic formatting rules and is currently implemented as a macro; for more precise control over the formatting process use the ~type~PushFormat() function.
128
-
129
- |*_int_* ~type~PushFloat(*_Type_* * +self+, *_double_* +value+)
130
- |
131
- Append string representation of the floating-point +value+ to string +self+.
132
-
133
- Return non-zero value on success and zero value on conversion error.
134
-
135
- NOTE: this convenience function applies generic formatting rules and is currently implemented as a macro; for more precise control over the formatting process use the ~type~PushFormat() function.
136
-
137
- |*_int_* ~type~PushPtr(*_Type_* * +self+, *_void*_* +value+)
138
- |
139
- Append string representation of the pointer +value+ to string +self+.
140
-
141
- Return non-zero value on success and zero value on conversion error.
142
-
143
- NOTE: this convenience function applies generic formatting rules and is currently implemented as a macro; for more precise control over the formatting process use the ~type~PushFormat() function.
144
-
145
- |*_void_* ~type~PushString(*_Type_* * +self+, *_Type_* * +from+)
146
- |
147
- Append a _copy_ of the contents of string +from+ to string +self+.
148
-
149
- |*_int_* ~type~PushFormat(*_Type_* * +self+, *_const char*_* format, ...);
150
- |
151
- Append the _?sprintf()_- formatted string to string +self+.
152
-
153
- Return non-zero value on successful formatting and zero value if the call to _?sprintf()_ failed.
154
- The latter usually happens due to the encoding error.
155
-
156
- This function tries to use the _vsnprintf()_ standard C function if possible and falls back to *unsafe* _vsprintf()_ function which is ought to be present in every ANSI-compliant standard C library.
157
- The former function is used on the platforms which are known to have it; the Autotools-compliant _HAVE_VSNPRINTF_ macro is also taken into consideration.
158
- _Note that the choice is obviously made at compile-time._
159
-
160
- If using the _vsnprintf()_ and the allocated buffer is not large enough this function continuously expands the buffer to eventually accommodate the resulting string.
161
- On the contrary, when the *unsafe* _vsprintf()_ is used, the buffer overrun causes this function to *abort()* in order to possible data corruption not to slip away uncaught.
162
-
163
- Current implementation operates on the heap-allocated buffer whose initial size is determined by the _AUTOC_BUFFER_SIZE_ macro.
164
- If not explicitly set it defaults to 4096 bytes.
165
- |===
166
-
167
- === Iteration over string's characters
168
-
169
- [cols=2*]
170
- |===
171
- |*_void_* ~it~Ctor(*_IteratorType_* * +it+, *_Type_* * +self+)
172
- |
173
- Create a new forward iterator +it+ on string +self+.
174
-
175
- NOTE: Previous contents of +it+ is overwritten.
176
-
177
- |*_void_* ~it~CtorEx(*_IteratorType_* * +it+, *_Type_* * +self+, *_int_* +forward+)
178
- |
179
- Create a new iterator +it+ on string +self+.
180
- Non-zero value of +forward+ specifies a forward iterator, zero value specifies a backward iterator.
181
-
182
- NOTE: Previous contents of +it+ is overwritten.
183
-
184
- |*_int_* ~it~Move(*_IteratorType_* * +it+)
185
- |
186
- Advance iterator position of +it+ *and* return non-zero value if new position is valid and zero value otherwise.
187
-
188
- |*_CharType_* ~it~Get(*_IteratorType_* * +it+)
189
- |
190
- Return a _copy_ of the character pointed to by the iterator +it+.
191
-
192
- WARNING: current position *must* be valid otherwise the behavior is undefined. See ~it~Move().
193
- |===
194
-
195
- =end
196
- class String < Type
197
-
198
- include Redirecting
199
-
200
- def char_type; :char end
201
-
202
- def char_type_ref; "#{char_type}*" end
203
-
204
- attr_reader :it_ref
205
-
206
- def initialize(type_name = :String, visibility = :public)
207
- super
208
- @it_ref = "#{it}*"
209
- @list = List.new(list, {:type => char_type_ref, :dtor => free}, :private) # List takes ownership over the strings put into it hence the custom element destructor
210
- initialize_redirectors
211
- @ctor = define_redirector(:ctor, Function::Signature.new([type_ref^:self, "const #{char_type_ref}"^:chars]))
212
- end
213
-
214
- # No default constructor provided
215
- def constructible?; false end
216
-
217
- def write_intf_types(stream)
218
- stream << %$
219
- /***
220
- **** #{type}<#{char_type}>
221
- ***/
222
- $ if public?
223
- [@list].each {|obj| obj.write_intf_types(stream)} # TODO : this should be handled by the entity dependencies system
224
- stream << %$
225
- typedef struct #{type} #{type};
226
- typedef struct #{it} #{it};
227
- struct #{type} {
228
- size_t size;
229
- union data {
230
- #{char_type_ref} string;
231
- #{@list.type} list;
232
- } data;
233
- int is_list;
234
- };
235
- struct #{it} {
236
- #{type_ref} string;
237
- int index, forward;
238
- };
239
- $
240
- end
241
-
242
- def write_intf_decls(stream, declare, define)
243
- super
244
- write_redirectors(stream, declare, define)
245
- stream << %$
246
- #include <string.h>
247
- #ifdef AUTOC_BUFFER_SIZE
248
- #define #{_bufferSize} AUTOC_BUFFER_SIZE
249
- #else
250
- #define #{_bufferSize} 4096 /* Stay in sync with the documentation! */
251
- #endif
252
- #define #{join}(self) if(self->is_list) #{_join}(self);
253
- #define #{split}(self) if(!self->is_list) #{_split}(self);
254
- #{declare} void #{_join}(#{type_ref});
255
- #{declare} void #{_split}(#{type_ref});
256
- #{declare} #{ctor.declaration};
257
- #{declare} #{dtor.declaration};
258
- #{declare} #{copy.declaration};
259
- #{declare} void #{copyRange}(#{type_ref}, #{type_ref}, size_t, size_t);
260
- #{declare} #{equal.declaration};
261
- #{declare} #{identify.declaration};
262
- #define #{empty}(self) (#{size}(self) == 0)
263
- #{define} size_t #{size}(#{type_ref} self) {
264
- #{assert}(self);
265
- /* #{join}(self); assuming the changes to the contents are reflected in the size */
266
- #ifndef NDEBUG
267
- /* Type invariants which must hold true */
268
- if(!self->is_list) #{assert}(self->size == strlen(self->data.string));
269
- /* TODO self->is_list case */
270
- #endif
271
- return self->size;
272
- }
273
- #{define} int #{within}(#{type_ref} self, size_t index) {
274
- #{assert}(self);
275
- /* Omitting excessive call to #{join}() */
276
- return index < #{size}(self);
277
- }
278
- #{define} #{char_type} #{get}(#{type_ref} self, size_t index) {
279
- #{assert}(self);
280
- #{assert}(#{within}(self, index));
281
- #{join}(self);
282
- return self->data.string[index];
283
- }
284
- #{define} void #{set}(#{type_ref} self, size_t index, #{char_type} value) {
285
- #{assert}(self);
286
- #{assert}(#{within}(self, index));
287
- #{join}(self);
288
- self->data.string[index] = value;
289
- }
290
- #{define} const #{char_type_ref} #{chars}(#{type_ref} self) {
291
- #{assert}(self);
292
- #{join}(self);
293
- return self->data.string;
294
- }
295
- #{declare} int #{pushFormat}(#{type_ref}, const char*, ...);
296
- #{declare} void #{pushChars}(#{type_ref}, const #{char_type_ref});
297
- #{declare} void #{pushString}(#{type_ref}, #{type_ref});
298
- #define #{pushChar}(self, c) #{pushFormat}(self, "%c", (char)(c))
299
- #define #{pushInt}(self, i) #{pushFormat}(self, "%d", (int)(i))
300
- #define #{pushFloat}(self, f) #{pushFormat}(self, "%e", (double)(f))
301
- #define #{pushPtr}(self, p) #{pushFormat}(self, "%p", (void*)(p))
302
- #define #{itCtor}(self, type) #{itCtorEx}(self, type, 1)
303
- #{define} void #{itCtorEx}(#{it_ref} self, #{type_ref} string, int forward) {
304
- #{assert}(self);
305
- #{assert}(string);
306
- self->string = string;
307
- self->forward = forward;
308
- self->index = forward ? -1 : #{size}(string);
309
- }
310
- #{define} int #{itMove}(#{it_ref} self) {
311
- #{assert}(self);
312
- if(self->forward) ++self->index; else --self->index;
313
- return #{within}(self->string, self->index);
314
- }
315
- #{define} #{char_type} #{itGet}(#{it_ref} self) {
316
- #{assert}(self);
317
- return #{get}(self->string, self->index);
318
- }
319
- $
320
- end
321
-
322
- def write_impls(stream, define)
323
- super
324
- [@list].each {|obj|
325
- obj.write_intf_decls(stream, static, inline)
326
- obj.write_impls(stream, static)
327
- }
328
- stream << %$
329
- #include <stdio.h>
330
- #include <stdarg.h>
331
- #undef AUTOC_VSNPRINTF
332
- #if defined(_MSC_VER)
333
- #define AUTOC_VSNPRINTF _vsnprintf
334
- #elif defined(__DMC__) || defined (__LCC__)
335
- #define AUTOC_VSNPRINTF vsnprintf
336
- #elif defined(HAVE_VSNPRINTF) || __STDC_VERSION__ >= 199901L /* Be Autotools-friendly, C99 must have snprintf() */
337
- #define AUTOC_VSNPRINTF vsnprintf
338
- #endif
339
- #ifndef AUTOC_VSNPRINTF
340
- /* #warning Using unsafe vsprintf() function */
341
- #endif
342
- #{define} void #{_join}(#{type_ref} self) {
343
- #{@list.it} it;
344
- #{char_type_ref} string;
345
- size_t* size; /* size sizes cache to avoid excessive calls to strlen() */
346
- size_t i, start = 0, total = 0;
347
- #{assert}(self);
348
- #{assert}(self->is_list);
349
- if(!#{@list.empty}(&self->data.list)) {
350
- size = (size_t*)malloc(#{@list.size}(&self->data.list)*sizeof(size_t)); #{assert}(size);
351
- #{@list.itCtor}(&it, &self->data.list);
352
- for(i = 0; #{@list.itMove}(&it); ++i) {
353
- total += (size[i] = strlen(#{@list.itGet}(&it)));
354
- }
355
- string = (#{char_type_ref})#{malloc}((total + 1)*sizeof(#{char_type})); #{assert}(string);
356
- #{@list.itCtor}(&it, &self->data.list);
357
- /* List is a LIFO structure therefore merging should be performed from right to left */
358
- i = 0; start = total;
359
- while(#{@list.itMove}(&it)) {
360
- start -= size[i];
361
- memcpy(&string[start], #{@list.itGet}(&it), size[i]*sizeof(#{char_type}));
362
- ++i;
363
- }
364
- string[total] = '\\0';
365
- #{free}(size);
366
- } else {
367
- string = (#{char_type_ref})#{calloc}(1, sizeof(#{char_type})); #{assert}(string);
368
- }
369
- #{@list.dtor}(&self->data.list);
370
- self->size = total;
371
- self->data.string = string;
372
- self->is_list = 0;
373
- }
374
- #{define} void #{_split}(#{type_ref} self) {
375
- #{@list.type} list;
376
- #{assert}(self);
377
- #{assert}(!self->is_list);
378
- #{@list.ctor}(&list);
379
- #{@list.push}(&list, self->data.string);
380
- /* self->size = strlen(self->data.string); not needed since the size shouldn't have changed */
381
- #{assert}(self->size == strlen(self->data.string));
382
- self->data.list = list;
383
- self->is_list = 1;
384
- }
385
- #{define} #{ctor.definition} {
386
- #{assert}(self);
387
- if(chars) {
388
- size_t nbytes;
389
- self->size = strlen(chars);
390
- nbytes = (self->size + 1)*sizeof(#{char_type});
391
- self->data.string = (#{char_type_ref})#{malloc}(nbytes); #{assert}(self->data.string);
392
- memcpy(self->data.string, chars, nbytes);
393
- self->is_list = 0;
394
- } else {
395
- /* NULL argument is permitted and corresponds to empty string */
396
- self->size = 0;
397
- #{@list.ctor}(&self->data.list);
398
- self->is_list = 1;
399
- }
400
- }
401
- #{define} #{dtor.definition} {
402
- #{assert}(self);
403
- if(self->is_list) #{@list.dtor}(&self->data.list); else #{free}(self->data.string);
404
- }
405
- #{define} #{copy.definition} {
406
- #{assert}(src);
407
- #{assert}(dst);
408
- #{assert}(src != dst);
409
- #{ctor}(dst, #{chars}(src));
410
- }
411
- #{define} void #{copyRange}(#{type_ref} dst, #{type_ref} src, size_t first, size_t last) {
412
- size_t size;
413
- #{char_type_ref} string;
414
- #{assert}(src);
415
- #{assert}(src != dst);
416
- #{assert}(first <= last);
417
- #{assert}(#{within}(src, first));
418
- #{assert}(#{within}(src, last));
419
- size = last - first + 1;
420
- string = (#{char_type_ref})#{malloc}((size + 1)*sizeof(#{char_type})); #{assert}(string);
421
- memcpy(string, &#{chars}(src)[first], size*sizeof(#{char_type}));
422
- string[size] = '\\0';
423
- #{ctor}(dst, string);
424
- #{free}(string);
425
- }
426
- #{define} #{equal.definition} {
427
- #{assert}(lt);
428
- #{assert}(rt);
429
- return strcmp(#{chars}(lt), #{chars}(rt)) == 0;
430
- }
431
- #{define} #{identify.definition} {
432
- size_t index, result = 0;
433
- #{assert}(self);
434
- #{join}(self);
435
- for(index = 0; index < #{size}(self); ++index) {
436
- result ^= self->data.string[index];
437
- result = AUTOC_RCYCLE(result);
438
- }
439
- return result;
440
- }
441
- #{define} int #{pushFormat}(#{type_ref} self, const char* format, ...) {
442
- va_list args;
443
- char* buffer;
444
- int i, c, buffer_size = #{_bufferSize};
445
- #{assert}(self);
446
- #{assert}(format);
447
- do {
448
- buffer = (char*)#{malloc}(buffer_size*sizeof(char)); #{assert}(buffer);
449
- va_start(args, format);
450
- #ifdef AUTOC_VSNPRINTF
451
- i = AUTOC_VSNPRINTF(buffer, buffer_size, format, args);
452
- #else
453
- i = vsprintf(buffer, format, args);
454
- if(i >= buffer_size) #{abort}();
455
- /* Since vsprintf() can not truncate its output this means the buffer overflow and
456
- there is no guarantee that some useful data is not corrupted so its better
457
- to crash right here than to let the corruption slip away uncaught */
458
- #endif
459
- c = (i > 0 && !(i < buffer_size));
460
- if(i > 0 && !c) #{pushChars}(self, buffer);
461
- va_end(args);
462
- #{free}(buffer);
463
- buffer_size *= 2;
464
- } while(c);
465
- return i >= 0;
466
- }
467
- #{define} void #{pushChars}(#{type_ref} self, const #{char_type_ref} chars) {
468
- #{char_type_ref} string;
469
- size_t size, nbytes;
470
- #{assert}(self);
471
- #{assert}(chars);
472
- #{split}(self);
473
- size = strlen(chars);
474
- nbytes = (size + 1)*sizeof(#{char_type});
475
- string = (#{char_type_ref})#{malloc}(nbytes); #{assert}(string);
476
- memcpy(string, chars, nbytes);
477
- #{@list.push}(&self->data.list, string);
478
- self->size += size;
479
- }
480
- #{define} void #{pushString}(#{type_ref} self, #{type_ref} from) {
481
- #{assert}(self);
482
- #{assert}(from);
483
- #{pushChars}(self, #{chars}(from));
484
- }
485
- #undef AUTOC_SNPRINTF
486
- $
487
- end
488
-
489
- end # String
490
-
491
-
492
- end # AutoC