![]() ![]() | ![]() |
1.1 snw 1: /*
2: * *
3: * * *
4: * * *
5: * ***************
6: * * * * *
7: * * MUMPS *
8: * * * * *
9: * ***************
10: * * *
11: * * *
12: * *
13: *
14: * mlib.h
15: * Function prototypes, structs, and macros for FreeM
16: * binding library
17: *
18: *
1.2 ! snw 19: * Author: Serena Willis <snw@coherent-logic.com>
1.1 snw 20: * Copyright (C) 1998 MUG Deutschland
21: * Copyright (C) 2020 Coherent Logic Development LLC
22: *
23: *
24: * This file is part of FreeM.
25: *
26: * FreeM is free software: you can redistribute it and/or modify
27: * it under the terms of the GNU Affero Public License as published by
28: * the Free Software Foundation, either version 3 of the License, or
29: * (at your option) any later version.
30: *
31: * FreeM is distributed in the hope that it will be useful,
32: * but WITHOUT ANY WARRANTY; without even the implied warranty of
33: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
34: * GNU Affero Public License for more details.
35: *
36: * You should have received a copy of the GNU Affero Public License
37: * along with FreeM. If not, see <https://www.gnu.org/licenses/>.
38: *
39: **/
40:
41: #include <stdlib.h>
42:
43: #include "mpsdef.h"
44: #include "freem.h"
45: #include "init.h"
46: #include "transact.h"
47: #include "version.h"
48:
49: #include <stdio.h>
50: #include <ctype.h>
51: #include <string.h>
52: #include <errno.h>
53:
54: /* the following function prototypes are not in freem.h, as they are intended to be
55: * private and internal to mlib.c
56: */
57: void mref2str(freem_ref_t *ref, char *key);
58: void freem_release_io(void);
59: void freem_return_io(void);
60:
61: extern int xecline(int typ);
62:
63: void mref2str(freem_ref_t *ref, char *key)
64: {
65: register int i;
66:
67: for (i = 0; i < 256; i++) key[i] = NUL;
68:
69: switch (ref->reftype) {
70:
71: case MREF_RT_LOCAL: /* do nothing for locals, as they have no sigil */
72: break;
73:
74: case MREF_RT_GLOBAL:
75: strcat (key, "^");
76: break;
77:
78: case MREF_RT_SSVN:
79: strcat (key, "^$");
80: break;
81:
82: }
83:
84: if (ref->subscript_count > 0) {
85: strcat (key, ref->name);
86: strcat (key, "\202");
87:
88: for (i = 0; i < ref->subscript_count; i++) {
89: strcat (key, ref->subscripts[i]);
90:
91: if (i < ref->subscript_count - 1) strcat (key, "\202");
92: }
93: }
94: else {
95: strcat (key, ref->name);
96: }
97:
98: // ends with EOL
99: strncat (key, "\201", 2);
100:
101: }
102:
103: pid_t freem_init(char *environment_name, char *namespace_name)
104: {
105: int ierr_sav;
106: int errno_sav;
107:
108: libflag = TRUE;
109: noclear = TRUE;
110: frm_filter = TRUE;
111: direct_mode = FALSE;
112:
113: errno = 0;
114:
115: strcpy (shm_env, environment_name);
116: snprintf (config_file, 4096, "%s/freem/%s/freem.conf", SYSCONFDIR, shm_env);
117:
118: init (namespace_name);
119: ierr_sav = ierr;
120: errno_sav = errno;
121:
122: freem_release_io ();
123:
124:
125: if (ierr_sav > OK || errno_sav != 0) {
126: return -1;
127: }
128: else {
129: return pid;
130: }
131: }
132:
133: short freem_version(char *result)
134: {
135: strncpy (result, FREEM_VERSION_ID, 255);
136:
137: return OK;
138: }
139:
140: short freem_get(freem_ref_t *ref)
141: {
142: char key[256];
143: char result[256];
144:
145: freem_return_io ();
146:
147: // set up the EOL-delimited string
148: mref2str (ref, key);
149:
150: switch (ref->reftype) {
151:
152: case MREF_RT_LOCAL:
153: // call into the symbol table to get the local var
154: symtab (get_sym, key, result);
155: ref->status = merr ();
156: break;
157:
158: case MREF_RT_GLOBAL:
159: // call into the FreeM global handler; result in &result
160: global (get_sym, key, result);
161: ref->status = merr ();
162: break;
163:
164: case MREF_RT_SSVN:
165: // call into the SSVN code
166: ssvn (get_sym, key, result);
167: ref->status = merr ();
168: break;
169:
170: }
171:
172: stcnv_m2c (result);
173:
174: strncpy (ref->value, result, 255);
175:
176: freem_release_io ();
177:
178: return ref->status;
179: }
180:
181: short freem_set(freem_ref_t *ref)
182: {
183: char key[256];
184: char data[256];
185:
186: freem_return_io ();
187:
188: mref2str (ref, key);
189:
190: snprintf (data, 254, "%s\201", ref->value);
191:
192: switch (ref->reftype) {
193:
194: case MREF_RT_LOCAL:
195: // call into the symbol table to set the local var
196: symtab (set_sym, key, data);
197: ref->status = merr ();
198: break;
199:
200: case MREF_RT_GLOBAL:
201: // call into the FreeM global handler; data in &data
202: global (set_sym, key, data);
203: ref->status = merr ();
204: break;
205:
206: case MREF_RT_SSVN:
207: // call into the SSVN code
208: ssvn (set_sym, key, data);
209: ref->status = merr ();
210: break;
211:
212: }
213:
214: freem_release_io ();
215:
216:
217: return ref->status;
218: }
219:
220: short freem_kill(freem_ref_t *ref)
221: {
222: char key[256];
223: char result[256];
224:
225: mref2str (ref, key);
226:
227: freem_return_io ();
228:
229: switch (ref->reftype) {
230:
231: case MREF_RT_LOCAL:
232: // call into the symbol table to kill the local var
233: symtab (kill_sym, key, result);
234: ref->status = merr ();
235: break;
236:
237: case MREF_RT_GLOBAL:
238: // call into the FreeM global handler; result in &result
239: global (kill_sym, key, result);
240: ref->status = merr ();
241: break;
242:
243: case MREF_RT_SSVN:
244: // call into the SSVN code
245: ssvn (kill_sym, key, result);
246: ref->status = merr ();
247: break;
248:
249: }
250:
251: stcnv_m2c (result);
252:
253: strncpy (ref->value, result, 255);
254:
255: freem_release_io ();
256:
257:
258: return ref->status;
259: }
260:
261: short freem_data(freem_ref_t *ref)
262: {
263: char key[256];
264: char result[256];
265:
266: mref2str (ref, key);
267:
268: freem_return_io ();
269:
270: switch (ref->reftype) {
271:
272: case MREF_RT_LOCAL:
273: // call into the symbol table
274: symtab (dat, key, result);
275: ref->status = merr ();
276: break;
277:
278: case MREF_RT_GLOBAL:
279: // call into the FreeM global handler; result in &result
280: global (dat, key, result);
281: ref->status = merr ();
282: break;
283:
284: case MREF_RT_SSVN:
285: // call into the SSVN code
286: ssvn (dat, key, result);
287: ref->status = merr ();
288: break;
289:
290: }
291:
292: stcnv_m2c (result);
293:
294: strncpy (ref->value, result, 255);
295:
296: freem_release_io ();
297:
298: return ref->status;
299: }
300:
301: short freem_order(freem_ref_t *ref)
302: {
303: char key[256];
304: char result[256];
305:
306: mref2str (ref, key);
307:
308: freem_return_io ();
309:
310: switch (ref->reftype) {
311:
312: case MREF_RT_LOCAL:
313: // call into the symbol table for $ORDER
314: symtab (fra_order, key, result);
315: ref->status = merr ();
316: break;
317:
318: case MREF_RT_GLOBAL:
319: // call into the FreeM global handler; result in &result
320: global (fra_order, key, result);
321: ref->status = merr ();
322: break;
323:
324: case MREF_RT_SSVN:
325: // call into the SSVN code
326: ssvn (fra_order, key, result);
327: ref->status = merr ();
328: break;
329:
330: }
331:
332: stcnv_m2c (result);
333:
334: strncpy (ref->value, result, 255);
335: freem_release_io ();
336:
337: return ref->status;
338: }
339:
340: short freem_query(freem_ref_t *ref)
341: {
342: char key[256];
343: char result[256];
344:
345: mref2str (ref, key);
346:
347: freem_return_io ();
348:
349: switch (ref->reftype) {
350:
351: case MREF_RT_LOCAL:
352: // call into the symbol table
353: symtab (fra_query, key, result);
354: ref->status = merr ();
355: break;
356:
357: case MREF_RT_GLOBAL:
358: // call into the FreeM global handler; result in &result
359: global (fra_query, key, result);
360: ref->status = merr ();
361: break;
362:
363: case MREF_RT_SSVN:
364: // call into the SSVN code
365: ssvn (fra_query, key, result);
366: ref->status = merr ();
367: break;
368:
369: }
370:
371: stcnv_m2c (result);
372:
373: strncpy (ref->value, result, 255);
374:
375: freem_release_io ();
376:
377: return ref->status;
378: }
379:
380: short freem_lock(freem_ref_t *ref, long lck_timeout)
381: {
382: char key[256];
383: char buf[256];
384:
385: mref2str (ref, key);
386: snprintf (buf, 255, "+%s\201", key);
387:
388: freem_return_io ();
389:
390: lock (buf, lck_timeout, 'l');
391: ref->status = merr ();
392:
393: freem_release_io ();
394:
395: return ref->status;
396: }
397:
398: short freem_unlock(freem_ref_t *ref, long lck_timeout)
399: {
400: char key[256];
401: char buf[256];
402:
403: mref2str (ref, key);
404: snprintf (buf, 255, "-%s\201", key);
405:
406: freem_return_io ();
407:
408: lock (buf, lck_timeout, 'l');
409: ref->status = merr ();
410:
411: freem_release_io ();
412:
413: return ref->status;
414: }
415:
416: short freem_tstart(char *tp_id, short serial, short restartable, char **sym_save)
417: {
418: return tp_tstart (tp_id, serial, restartable, sym_save);
419: }
420:
421: short freem_trestart(void)
422: {
423: return tp_trestart ();
424: }
425:
426: short freem_trollback(int tp_levels)
427: {
428: return tp_trollback (tp_levels);
429: }
430:
431: short freem_tcommit(void)
432: {
433: return tp_tcommit ();
434: }
435:
436: int freem_tlevel(void)
437: {
438: return tp_level;
439: }
440:
441: short freem_function(freem_ent_t *ent)
442: {
443: register int i;
444: char buf[STRLEN] = {0};
445: char tmp[STRLEN] = {0};
446: char tmp1[STRLEN] = {0};
447:
448: freem_return_io ();
449:
450: if (ent->argument_count > 0) {
451: snprintf (buf, STRLEN - 1, "S %%ZFRMRTN=$$%s(", ent->name);
452:
453: for (i = 0; i < ent->argument_count; i++) {
454:
455: strncpy (tmp1, ent->arguments[i], STRLEN - 1);
456: stcnv_c2m (tmp1);
457:
458: if (znamenumeric (tmp1) == TRUE) {
459: /* arguments that are canonical MUMPS numbers don't get quoted */
460: snprintf (tmp, STRLEN - 1, "%s", ent->arguments[i]);
461: }
462: else {
463: /* string arguments do get quoted */
464: snprintf (tmp, STRLEN - 1, "\"%s\"", ent->arguments[i]);
465: }
466:
467: strcat (buf, tmp);
468:
469: if(i < ent->argument_count - 1) {
470: strcat (buf, ",");
471: }
472: }
473:
474: strcat (buf, ")");
475: }
476: else {
477: snprintf (buf, STRLEN - 1, "S %%ZFRMRTN=$$%s", ent->name);
478: }
479:
480: stcnv_c2m (buf);
481: sprintf (tmp, "%%ZFRMXEC\201");
482: symtab (set_sym, tmp, buf);
483: snprintf ((char *) ztrap, 12, "^%%ZFRMXEC\201");
484:
485: xecline (3);
486: ent->status = merr ();
487:
488: tmp[0] = NUL;
489: ent->value[0] = NUL;
490:
491: snprintf (tmp, 255, "%%ZFRMRTN\201");
492: symtab (get_sym, tmp, ent->value);
493: stcnv_m2c (ent->value);
494:
495: freem_release_io ();
496:
497: return ent->status;
498: }
499:
500: short freem_procedure(freem_ent_t *ent)
501: {
502: register int i;
503: char buf[STRLEN] = {0};
504: char tmp[STRLEN] = {0};
505: char tmp1[STRLEN] = {0};
506:
507: freem_return_io ();
508:
509: if (ent->argument_count > 0) {
510: snprintf (buf, STRLEN - 1, "DO %s(", ent->name);
511:
512: for (i = 0; i < ent->argument_count; i++) {
513:
514: strncpy (tmp1, ent->arguments[i], STRLEN - 1);
515: stcnv_c2m (tmp1);
516:
517: if (znamenumeric (tmp1) == TRUE) {
518: /* arguments that are canonical MUMPS numbers don't get quoted */
519: snprintf (tmp, STRLEN - 1, "%s", ent->arguments[i]);
520: }
521: else {
522: /* string arguments do get quoted */
523: snprintf (tmp, STRLEN - 1, "\"%s\"", ent->arguments[i]);
524: }
525:
526: strcat (buf, tmp);
527:
528: if(i < ent->argument_count - 1) {
529: strcat (buf, ",");
530: }
531: }
532:
533: strcat (buf, ")");
534: }
535: else {
536: snprintf (buf, STRLEN - 1, "DO %s", ent->name);
537: }
538:
539: stcnv_c2m (buf);
540: sprintf (tmp, "%%ZFRMXEC\201");
541: symtab (set_sym, tmp, buf);
542:
543:
544: snprintf ((char *) ztrap, 12, "^%%ZFRMXEC\201");
545:
546:
547: xecline (3);
548: ent->status = merr ();
549:
550: tmp[0] = NUL;
551: ent->value[0] = NUL;
552:
553: freem_release_io ();
554:
555: return ent->status;
556: }
557:
558: short freem_errmsg(int code, char *msg)
559: {
560: char emtmp[256];
561: char cvtmp[256];
562:
563: if (code > MAXERR || code < 0) {
564: return FALSE;
565: }
566:
567: stcpy (cvtmp, errmes[code]);
568: stcnv_m2c (cvtmp);
569:
570: merr_num_to_code (code, emtmp);
571:
572: sprintf (msg, "%s: %s", emtmp, cvtmp);
573:
574: return TRUE;
575: }
576:
577: void freem_release_io(void)
578: {
579: set_io (UNIX);
580: }
581:
582: void freem_return_io(void)
583: {
584: set_io (MUMPS);
585: }