1 | |
---|
2 | #include <fstream.hfa> |
---|
3 | #include <collections/string.hfa> |
---|
4 | #include <collections/string_res.hfa> |
---|
5 | #include <stdio.h> |
---|
6 | |
---|
7 | // No-op manipulators. |
---|
8 | // Temporary hack while there are two code paths in the string implementation. |
---|
9 | // (One for reading plain strings, the other for reading via a manipulator.) |
---|
10 | // The test cases that use plainjane(-) are exercising the via-manipulator code path, |
---|
11 | // just with trivial manipulation. |
---|
12 | static _Istream_Sstr plainjane( string & s ) { return (_Istream_Sstr)@{ s, {{0p}, -1, {.flags.rwd : false}} }; } |
---|
13 | static _Istream_Rstr plainjane( string_res & s ) { return (_Istream_Rstr)@{ &s, {{0p}, -1, {.flags.rwd : false}} }; } |
---|
14 | |
---|
15 | static void forceStringHeapFreeSpaceTo(int desiredSize) { |
---|
16 | for (1_000_000) { |
---|
17 | string x = "a"; |
---|
18 | (void)x; |
---|
19 | if (desiredSize == DEBUG_string_bytes_avail_until_gc(DEBUG_string_heap())) return; |
---|
20 | } |
---|
21 | sout | "Unable to force size" | desiredSize | "in 1,000,000 tries"; |
---|
22 | } |
---|
23 | |
---|
24 | int main() { |
---|
25 | // These "pre" cases deal with issues analogous to the "pre" cases of io/manipulatorsInput. |
---|
26 | // The acceptance criterion is simpler but driving the cases is harder. |
---|
27 | // The tests just read strings and echo what they read; acceptance of simple echoing assures |
---|
28 | // no spurious splitting merging. |
---|
29 | // The lengths of the strings are chosen to match white-box knowledge of when the string layer |
---|
30 | // has tor drive the cstring layer through a second iteration: |
---|
31 | // - for no-manip, lengths are near the room at end of string heap |
---|
32 | // (chosen target size of 9 showed the original bug on preS2, aligned with the other cases) |
---|
33 | // - for manip, lengths are near the auxiliary buffer size of 128 |
---|
34 | // Only first case repeats for string_res; rest run only from the passthru string layer. |
---|
35 | // Similarly, the manipulator breadth isn't checked at the cstring layer either. |
---|
36 | { |
---|
37 | // S: string, no manipulator |
---|
38 | void echoTillX(const char * casename) { |
---|
39 | string s; |
---|
40 | // loop assumes behaviour not tested until main-case #15: |
---|
41 | // on reading nothing, the prior string value is left alone |
---|
42 | do { |
---|
43 | s = ""; |
---|
44 | forceStringHeapFreeSpaceTo(9); |
---|
45 | sin | s; |
---|
46 | sout | casename | s; |
---|
47 | } while ( size(s) > 0 && s[size(s)-1] != 'x' ); |
---|
48 | } |
---|
49 | echoTillX("preS1"); |
---|
50 | echoTillX("preS2"); |
---|
51 | echoTillX("preS3"); |
---|
52 | echoTillX("preS4"); |
---|
53 | } |
---|
54 | { |
---|
55 | // SMN: string, manipulator for no-op |
---|
56 | void echoTillX(const char * casename) { |
---|
57 | string s; |
---|
58 | do { |
---|
59 | s = ""; |
---|
60 | sin | plainjane( s ); |
---|
61 | sout | casename | s; |
---|
62 | } while ( size(s) > 0 && s[size(s)-1] != 'x' ); |
---|
63 | } |
---|
64 | echoTillX("preSMN1"); |
---|
65 | echoTillX("preSMN2"); |
---|
66 | echoTillX("preSMN3"); |
---|
67 | echoTillX("preSMN4"); |
---|
68 | } |
---|
69 | { |
---|
70 | // RMN: string_res, manipulator for no-op |
---|
71 | void echoTillX(const char * casename) { |
---|
72 | string_res s; |
---|
73 | do { |
---|
74 | s = ""; |
---|
75 | sin | plainjane( s ); |
---|
76 | sout | casename | s; |
---|
77 | } while ( size(s) > 0 && s[size(s)-1] != 'x' ); |
---|
78 | } |
---|
79 | echoTillX("preRMN1"); |
---|
80 | echoTillX("preRMN2"); |
---|
81 | echoTillX("preRMN3"); |
---|
82 | echoTillX("preRMN4"); |
---|
83 | } |
---|
84 | { |
---|
85 | // SMI: string, manipulator `incl` |
---|
86 | void echoTillX(const char * casename) { |
---|
87 | string s; |
---|
88 | do { |
---|
89 | s = ""; |
---|
90 | sin | skip("-\n"); |
---|
91 | sin | incl( ".:|# x", s ); |
---|
92 | sout | casename | " \"" | s | "\""; |
---|
93 | } while ( size(s) > 0 && s[size(s)-1] != 'x' ); |
---|
94 | } |
---|
95 | echoTillX("preSMI1"); |
---|
96 | echoTillX("preSMI2"); |
---|
97 | echoTillX("preSMI3"); |
---|
98 | echoTillX("preSMI4"); |
---|
99 | } |
---|
100 | { |
---|
101 | // SME: string, manipulator `excl` |
---|
102 | void echoTillX(const char * casename) { |
---|
103 | string s; |
---|
104 | do { |
---|
105 | s = ""; |
---|
106 | sin | skip("-\n"); |
---|
107 | sin | excl( "-\n", s ); |
---|
108 | sout | casename | " \"" | s | "\""; |
---|
109 | } while ( size(s) > 0 && s[size(s)-1] != 'x' ); |
---|
110 | } |
---|
111 | echoTillX("preSME1"); |
---|
112 | echoTillX("preSME2"); |
---|
113 | echoTillX("preSME3"); |
---|
114 | echoTillX("preSME4"); |
---|
115 | } |
---|
116 | sin | skip("-\n"); |
---|
117 | { |
---|
118 | // SMG: string, manipulator `getline` |
---|
119 | void echoTillX(const char * casename) { |
---|
120 | string s; |
---|
121 | do { |
---|
122 | s = ""; |
---|
123 | sin | getline( s ); |
---|
124 | sout | casename | s; |
---|
125 | } while ( size(s) > 0 && s[size(s)-1] != 'x' ); |
---|
126 | } |
---|
127 | echoTillX("preSMG1"); |
---|
128 | echoTillX("preSMG2"); |
---|
129 | echoTillX("preSMG3"); |
---|
130 | echoTillX("preSMG4"); |
---|
131 | } |
---|
132 | { |
---|
133 | // SMD: string, manipulator (`getline` with custom) delimiter |
---|
134 | void echoTillX(const char * casename) { |
---|
135 | string s; |
---|
136 | do { |
---|
137 | s = ""; |
---|
138 | sin | getline( s, '@' ); |
---|
139 | sout | casename | s; |
---|
140 | } while ( size(s) > 0 && s[size(s)-1] != 'x' ); |
---|
141 | sin | skip(" \n"); |
---|
142 | } |
---|
143 | echoTillX("preSMD1"); |
---|
144 | echoTillX("preSMD2"); |
---|
145 | echoTillX("preSMD3"); |
---|
146 | echoTillX("preSMD4"); |
---|
147 | } |
---|
148 | // Keep harmonized with io/manipulatorsInput. |
---|
149 | { |
---|
150 | string s = "yyyyyyyyyyyyyyyyyyyy"; |
---|
151 | char sk[] = "abc"; |
---|
152 | sin | "abc " | skip( sk ) | skip( 5 ); sout | "1" | s; |
---|
153 | sin | s; sout | "2" | s; |
---|
154 | sin | ignore( s ); sout | "3" | s; |
---|
155 | sin | wdi( 8, s ); sout | "4" | s; |
---|
156 | sin | ignore( wdi( 8, s ) ); sout | "5" | s; |
---|
157 | |
---|
158 | sin | incl( "abc", s ); sout | "6" | s; |
---|
159 | sin | excl( "abc", s ); sout | "7" | s; |
---|
160 | sin | ignore( incl( "abc", s ) ); sout | "8" | s; |
---|
161 | sin | ignore( excl( "abc", s ) ); sout | "9" | s; |
---|
162 | sin | incl( "abc", wdi( 8, s ) ); sout | "10" | s; |
---|
163 | sin | excl( "abc", wdi( 8, s ) ); sout | "11" | s; |
---|
164 | sin | ignore( incl( "abc", wdi( 8, s ) ) ); sout | "12" | s; |
---|
165 | sin | ignore( excl( "abc", wdi( 8, s ) ) ); sout | "13" | s; |
---|
166 | sin | nl; |
---|
167 | |
---|
168 | s = "q"; |
---|
169 | sin | incl( "abc", s ); sout | "14" | s; |
---|
170 | s = "q"; |
---|
171 | sin | excl( "u", s ); sout | "15" | s; |
---|
172 | sin | skip( "u" ) | nl; |
---|
173 | |
---|
174 | sin | getline( s ); sout | "16" | s; |
---|
175 | sin | getline( s, '%' ) | nl; sout | "17" | s; |
---|
176 | sin | ignore( getline( s, '%' ) ) | nl; sout | "18" | s; |
---|
177 | |
---|
178 | sin | quoted( s ); sout | "19" | s; |
---|
179 | sin | quoted( s, '\'' ); sout | "20" | s; |
---|
180 | sin | quoted( s, '{', '}' ); sout | "21" | s; |
---|
181 | sin | quoted( s, 'X', 'Y' ); sout | "22" | s; |
---|
182 | sin | ignore( quoted( s, 'X', 'Y' ) ); sout | "23" | s; |
---|
183 | sin | nl; |
---|
184 | } |
---|
185 | // Full repeat on string_res layer assures the full manipulator vocabulary is supported there. |
---|
186 | { |
---|
187 | string_res s = "yyyyyyyyyyyyyyyyyyyy"; |
---|
188 | char sk[] = "abc"; |
---|
189 | sin | "abc " | skip( sk ) | skip( 5 ); sout | "1" | s; |
---|
190 | sin | s; sout | "2" | s; |
---|
191 | sin | ignore( s ); sout | "3" | s; |
---|
192 | sin | wdi( 8, s ); sout | "4" | s; |
---|
193 | sin | ignore( wdi( 8, s ) ); sout | "5" | s; |
---|
194 | |
---|
195 | sin | incl( "abc", s ); sout | "6" | s; |
---|
196 | sin | excl( "abc", s ); sout | "7" | s; |
---|
197 | sin | ignore( incl( "abc", s ) ); sout | "8" | s; |
---|
198 | sin | ignore( excl( "abc", s ) ); sout | "9" | s; |
---|
199 | sin | incl( "abc", wdi( 8, s ) ); sout | "10" | s; |
---|
200 | sin | excl( "abc", wdi( 8, s ) ); sout | "11" | s; |
---|
201 | sin | ignore( incl( "abc", wdi( 8, s ) ) ); sout | "12" | s; |
---|
202 | sin | ignore( excl( "abc", wdi( 8, s ) ) ); sout | "13" | s; |
---|
203 | sin | "\n"; |
---|
204 | |
---|
205 | s = "q"; |
---|
206 | sin | incl( "abc", s ); sout | "14" | s; |
---|
207 | s = "q"; |
---|
208 | sin | excl( "u", s ); sout | "15" | s; |
---|
209 | sin | skip( "u" ); |
---|
210 | sin | "\n"; |
---|
211 | sin | getline( s ); sout | "16" | s; |
---|
212 | sin | getline( s, '%' ) | nl; sout | "17" | s; |
---|
213 | sin | ignore( getline( s, '%' ) ) | nl; sout | "18" | s; |
---|
214 | |
---|
215 | sin | quoted( s ); sout | "19" | s; |
---|
216 | sin | quoted( s, '\'' ); sout | "20" | s; |
---|
217 | sin | quoted( s, '{', '}' ); sout | "21" | s; |
---|
218 | sin | quoted( s, 'X', 'Y' ); sout | "22" | s; |
---|
219 | sin | ignore( quoted( s, 'X', 'Y' ) ); sout | "23" | s; |
---|
220 | } |
---|
221 | } |
---|