rts-jit.c
1 //! @file jit.c
2 //! @author J. Marcel van der Veer
3 //
4 //! @section Copyright
5 //
6 // This file is part of VIF - vintage FORTRAN compiler.
7 // Copyright 2020-2025 J. Marcel van der Veer <algol68g@xs4all.nl>.
8 //
9 //! @section License
10 //
11 // This program is free software; you can redistribute it and/or modify it
12 // under the terms of the GNU General Public License as published by the
13 // Free Software Foundation; either version 3 of the License, or
14 // (at your option) any later version.
15 //
16 // This program is distributed in the hope that it will be useful, but
17 // WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
18 // or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
19 // more details. You should have received a copy of the GNU General Public
20 // License along with this program. If not, see <http://www.gnu.org/licenses/>.
21
22 //! @section Synopsis
23 //!
24 //! Just-in-time compilation of FORMAT strings.
25
26 #include <vif.h>
27
28 #define HEAP_SIZE 8192
29 #define MAX_ITEMS 256
30
31 static char heap[HEAP_SIZE];
32 static char *fmtarr[MAX_ITEMS];
33 static size_t heaptr, arrptr;
34
35 static char *jit_str_list (char *, char *, int *, int *);
36
37 static void jit_error (char *where, char *msg)
38 {
39 RECORD diag;
40 _srecordf (diag, "format compiler: %s", msg);
41 RTE (where, diag);
42 }
43
44 static char *alloc (char *where, char *elem)
45 {
46 size_t N = strlen (elem) + 1;
47 if ((heaptr + N) >= (HEAP_SIZE - 1)) {
48 jit_error (where, "heap overflow");
49 return NO_TEXT;
50 } else {
51 char *q = &heap[heaptr];
52 strcpy (q, elem);
53 heaptr += N;
54 return q;
55 }
56 }
57
58 static void add_mark (char *where, char *elem)
59 {
60 if (arrptr == (MAX_ITEMS - 1)) {
61 jit_error (where, "too many items");
62 } else {
63 fmtarr[arrptr ++] = elem;
64 }
65 }
66
67 static void add_item (char *where, char *elem)
68 {
69 if (arrptr == (MAX_ITEMS - 1)) {
70 jit_error (where, "too many items");
71 } else {
72 if (elem == NO_TEXT) {
73 fmtarr[arrptr ++] = NO_TEXT;
74 } else {
75 fmtarr[arrptr ++] = alloc (where, elem);
76 }
77 }
78 }
79
80 static void jit_elem (char *where, char *lex, int_4 *elems)
81 {
82 RECORD rstr, wstr;
83 int_4 width, digits, expwid;
84 if (LEQUAL ("a", lex)) {
85 if (sscanf (&lex[1], "%d", &width) != 1) {
86 _srecordf (rstr, "%%s");
87 _srecordf (wstr, "%%s");
88 } else if (width < 1) {
89 jit_error (where, lex);
90 _srecordf (rstr, "%%s");
91 _srecordf (wstr, "%%s");
92 } else {
93 _srecordf (rstr, "%%%ds", width);
94 _srecordf (wstr, "%%-%ds", width);
95 }
96 add_mark (where, FMT_CHAR);
97 (*elems)++;
98 } else if (LEQUAL ("i", lex)) {
99 if (sscanf (&lex[1], "%d", &width) != 1) {
100 jit_error (where, "expected width");
101 _srecordf (rstr, "%%d");
102 _srecordf (wstr, "%%d");
103 } else if (width < 1) {
104 jit_error (where, lex);
105 _srecordf (rstr, "%%d");
106 _srecordf (wstr, "%%d");
107 } else {
108 _srecordf (rstr, "%%%dd", width);
109 _srecordf (wstr, "%%%dd", width);
110 }
111 add_mark (where, FMT_INT);
112 (*elems)++;
113 } else if (LEQUAL ("l", lex)) {
114 if (sscanf (&lex[1], "%d", &width) != 1) {
115 jit_error (where, "expected width");
116 _srecordf (rstr, "%%s");
117 _srecordf (wstr, "%%s");
118 } else if (width < 1) {
119 jit_error (where, lex);
120 _srecordf (rstr, "%%s");
121 _srecordf (wstr, "%%s");
122 } else {
123 _srecordf (rstr, "%%%ds", width);
124 _srecordf (wstr, "%%-%ds", width);
125 }
126 add_mark (where, FMT_INT);
127 (*elems)++;
128 } else if (LEQUAL ("d", lex) || LEQUAL ("e", lex) || LEQUAL ("g", lex) || LEQUAL ("n", lex)) {
129 width = digits = expwid = 0;
130 if (sscanf (&lex[1], "%d.%d.%d", &width, &digits, &expwid) == 3) {
131 ;
132 } else if (sscanf (&lex[1], "%d.%d", &width, &digits) == 2) {
133 ;
134 } else {
135 jit_error (where, "expected width, decimals, [width]");
136 }
137 if (width < 0 || digits < 0) {
138 jit_error (where, lex);
139 }
140 // Reading a REAL - specify width only!
141 _srecordf (rstr, "%%%de", width);
142 // Writing a REAL - specify all.
143 if (tolower (lex[0]) == 'n') {
144 if (islower (lex[0])) {
145 _srecordf (wstr, "%%%d.%d.%dn", width, digits, expwid);
146 } else {
147 _srecordf (wstr, "%%%d.%d.%dN", width, digits, expwid);
148 }
149 } else {
150 if (islower (lex[0])) {
151 _srecordf (wstr, "%%%d.%d.%de", width, digits, expwid);
152 } else {
153 _srecordf (wstr, "%%%d.%d.%dE", width, digits, expwid);
154 }
155 }
156 add_mark (where, FMT_REAL);
157 (*elems)++;
158 } else if (LEQUAL ("f", lex)) {
159 sscanf (&lex[1], "%d.%d", &width, &digits);
160 if (sscanf (&lex[1], "%d.%d", &width, &digits) != 2) {
161 jit_error (where, "expected width, decimals");
162 _srecordf (rstr, "%%f");
163 _srecordf (wstr, "%%f");
164 } else if (width < 1 || digits < 0) {
165 jit_error (where, lex);
166 _srecordf (rstr, "%%f");
167 _srecordf (wstr, "%%f");
168 } else {
169 _srecordf (rstr, "%%%df", width);
170 _srecordf (wstr, "%%%d.%df", width, digits);
171 }
172 add_mark (where, FMT_REAL);
173 (*elems)++;
174 } else {
175 jit_error (where, lex);
176 rstr[0] = wstr[0] = '\0';
177 (*elems)++;
178 }
179 RECORD fstr;
180 _srecordf (fstr, "%s", rstr);
181 add_item (where, fstr);
182 _srecordf (fstr, "%s", wstr);
183 add_item (where, fstr);
184 }
185
186 static void jit_scale (char *where, int_4 N)
187 {
188 add_mark (where, FMT_TEXT);
189 add_item (where, "");
190 if (N == 0) {
191 add_item (where, "0");
192 } else if (N == 1) {
193 add_item (where, "1");
194 } else if (N == 2) {
195 add_item (where, "2");
196 } else if (N == 3) {
197 add_item (where, "3");
198 } else {
199 add_item (where, "1");
200 }
201 }
202
203 static void jit_x (char *where, int_4 N)
204 {
205 RECORD str;
206 str[0] = '\0';
207 for (int_4 k = 0; k < N && k < RECLN; k ++) {
208 bufcat (str, " ", RECLN);
209 }
210 add_mark (where, FMT_TEXT);
211 add_item (where, str);
212 add_item (where, str);
213 }
214
215 static void jit_nl (char *where, int_4 N)
216 {
217 RECORD str;
218 _srecordf (str, "\\n");
219 for (int_4 k = 0; k < N; k ++) {
220 add_mark (where, FMT_TEXT);
221 add_item (where, str);
222 add_item (where, str);
223 }
224 }
225
226 static void jit_text (char *where, int_4 N, char *lex)
227 {
228 for (int_4 k = 0; k < N; k ++) {
229 add_mark (where, FMT_TEXT);
230 add_item (where, lex);
231 add_item (where, lex);
232 }
233 }
234
235 static char *jit_str_list (char *where, char *fmt, int *nest, int *elems)
236 {
237 #define LETTER(ch) (tolower (fmt[0]) == ch)
238 if (fmt[0] == '(') {
239 fmt++;
240 } else {
241 jit_error (where, "expected '('");
242 }
243 while (fmt[0] != ')' && fmt[0] != '\0') {
244 if (fmt[0] == ',') {
245 fmt++;
246 }
247 while (isspace (fmt[0])) {
248 fmt++;
249 }
250 int_4 N;
251 if (!isdigit (fmt[0])) {
252 N = 1;
253 } else {
254 char *end;
255 N = strtol (fmt, &end, 10);
256 fmt = end;
257 }
258 if (LETTER ('p')) {
259 jit_scale (where, N);
260 (*elems)++;
261 fmt++;
262 } else if (LETTER ('x')) {
263 jit_x (where, N);
264 (*elems)++;
265 fmt++;
266 } else if (fmt[0] == '/') {
267 jit_nl (where, N);
268 (*elems)++;
269 fmt++;
270 } else if (LETTER ('h')) {
271 fmt++;
272 RECORD str;
273 RECCLR (str);
274 int_4 k = 0;
275 for (int_4 M = 0; M < N && fmt[0] != '\0'; M++) {
276 str[k++] = (fmt++)[0];
277 }
278 jit_text (where, 1, str);
279 (*elems)++;
280 } else if (fmt[0] == '\"') {
281 fmt++;
282 RECORD str;
283 RECCLR (str);
284 int_4 k = 0;
285 int_4 go_on = TRUE;
286 while (go_on) {
287 if (fmt[0] == '\0') {
288 go_on = FALSE;
289 } else if (fmt[0] == '"') {
290 if ((++fmt)[0] == '"') {
291 str[k++] = '"';
292 fmt++;
293 } else {
294 go_on = FALSE;
295 }
296 } else {
297 str[k++] = (fmt++)[0];
298 }
299 }
300 jit_text (where, N, str);
301 (*elems)++;
302 } else if (fmt[0] == '\'') {
303 fmt++;
304 RECORD str;
305 RECCLR (str);
306 int_4 k = 0;
307 int_4 go_on = TRUE;
308 while (go_on) {
309 if (fmt[0] == '\0') {
310 go_on = FALSE;
311 } else if (fmt[0] == '\'') {
312 if ((++fmt)[0] == '\'') {
313 str[k++] = '\'';
314 fmt++;
315 } else {
316 go_on = FALSE;
317 }
318 } else {
319 str[k++] = (fmt++)[0];
320 }
321 }
322 jit_text (where, N, str);
323 (*elems)++;
324 } else {
325 for (int_4 k = 0; k < N; k++) {
326 char *sav = fmt, *rtn = NO_TEXT;
327 if (fmt[0] == '(') {
328 (*nest)++;
329 rtn = jit_str_list (where, fmt, nest, elems);
330 } else if (fmt[0] == ')') {
331 break;
332 } else if (strchr ("adefgiln", tolower (fmt[0])) == NO_TEXT) {
333 jit_error (where, fmt);
334 } else {
335 RECORD lex;
336 RECCLR (lex);
337 char *p = lex;
338 do {
339 (p++)[0] = (fmt++)[0];
340 while (isdigit (fmt[0])) {
341 (p++)[0] = (fmt++)[0];
342 }
343 } while (fmt[0] == '.');
344 jit_elem (where, lex, elems);
345 }
346 if (k < N - 1) {
347 fmt = sav;
348 } else if (rtn != NO_TEXT) {
349 fmt = rtn;
350 }
351 }
352 }
353 }
354 if (fmt[0] != ')') {
355 jit_error (where, "expected ')'");
356 return fmt;
357 } else if (*nest > 0) {
358 (*nest)--;
359 }
360 return &fmt[1];
361 #undef LETTER
362 }
363
364 char **_vif_jit (char *where, char *arg)
365 {
366 int_4 nest = 0, elems = 0;
367 RECORD cpy;
368 RECCLR (cpy);
369 fmtarr[0] = NO_TEXT;
370 heaptr = 0;
371 arrptr = 0;
372 strcpy (cpy, arg);
373 char *fmt = (char *) cpy;
374 if (fmt[0] == '"') {
375 fmt++;
376 }
377 if (fmt[strlen (fmt) - 1] == '"') {
378 fmt[strlen (fmt) - 1] = '\0';
379 }
380 (void) jit_str_list (where, fmt, &nest, &elems);
381 if (nest != 0) {
382 jit_error (where, "unbalanced parentheses");
383 }
384 if (elems == 0) {
385 jit_error (where, "empty format");
386 }
387 add_item (where, NO_TEXT);
388 add_item (where, NO_TEXT);
389 add_item (where, NO_TEXT);
390 return fmtarr;
391 }
© 2002-2025 J.M. van der Veer (jmvdveer@xs4all.nl)
|