1 /**
2 *** XMLCATALOG command response program.
3 ***
4 *** See Copyright for the status of this software.
5 ***
6 *** Author: Patrick Monnerat <pm@datasphere.ch>, DATASPHERE S.A.
7 **/
8
9 #include <stdio.h>
10 #include <stdlib.h>
11 #include <string.h>
12 #include <qshell.h>
13
14
15 /* Variable-length string, with 16-bit length. */
16 typedef struct {
17 short len;
18 char string[5000];
19 } vary2;
20
21
22 /* Variable-length string, with 32-bit length. */
23 typedef struct {
24 int len;
25 char string[5000];
26 } vary4;
27
28
29 /* Multiple occurrence parameter list. */
30 #define paramlist(itemsize, itemtype) \
31 _Packed struct { \
32 short len; \
33 _Packed union { \
34 char _pad[itemsize]; \
35 itemtype param; \
36 } item[1]; \
37 }
38
39 /* Add element list structure. */
40 typedef struct {
41 short elcount; /* Element count (=3). */
42 paramlist(16, char) type; /* vary2(16). */
43 paramlist(256, char) origin; /* vary2(256). */
44 paramlist(256, char) replace; /* vary2(256). */
45 } addelement;
46
47 /* SGML add element list structure. */
48 typedef struct {
49 short elcount; /* Element count (=3). */
50 paramlist(256, char) catalog; /* vary2(256). */
51 paramlist(256, char) ident; /* vary2(256). */
52 } sgmladdelement;
53
54
55 /* Arguments from CL command. */
56 typedef struct {
57 char * pgm; /* Program name. */
58 vary2 * instmf; /* Input catalog file name. */
59 vary2 * kind; /* Catalog kind. */
60 vary2 * outstmf; /* Output catalog file name. */
61 vary2 * convert; /* Convert SGML to XML. */
62 vary2 * superupd; /* --no-super-update. */
63 vary2 * verbose; /* Verbose output. */
64 paramlist(256 + 2, vary2) * delete; /* Identifiers to delete. */
65 paramlist(2, unsigned short) * add; /* Items to add. */
66 paramlist(2, unsigned short) * sgmladd; /* SGML items to add. */
67 paramlist(256 + 2, vary2) * resolve; /* Identifiers to resolve. */
68 paramlist(5000 + 2, vary2) * catalog; /* Additional catalog files. */
69 } arguments;
70
71
72 /* Definition of QSHELL program. */
73 extern void qshell(vary4 * cmd);
74 #pragma linkage(qshell, OS)
75 #pragma map(qshell, "QSHELL/QZSHQSHC")
76
77 /* Macro to handle displacements. */
78 #define OFFSETBY(t, p, n) ((t *) (((char *) (p)) + (n)))
79
80
81 static void
vary4nappend(vary4 * dst,const char * src,size_t len)82 vary4nappend(vary4 * dst, const char * src, size_t len)
83
84 {
85 if (len > sizeof(dst->string) - dst->len)
86 len = sizeof(dst->string) - dst->len;
87
88 if (len) {
89 memcpy(dst->string + dst->len, src, len);
90 dst->len += len;
91 }
92 }
93
94
95 static void
vary4append(vary4 * dst,const char * src)96 vary4append(vary4 * dst, const char * src)
97
98 {
99 vary4nappend(dst, src, strlen(src));
100 }
101
102
103 static void
vary4arg(vary4 * dst,const char * arg)104 vary4arg(vary4 * dst, const char * arg)
105
106 {
107 vary4nappend(dst, " ", 1);
108 vary4append(dst, arg);
109 }
110
111
112 static void
vary4varg(vary4 * dst,vary2 * arg)113 vary4varg(vary4 * dst, vary2 * arg)
114
115 {
116 vary4nappend(dst, " ", 1);
117 vary4nappend(dst, arg->string, arg->len);
118 }
119
120
121 static void
vary4vescape(vary4 * dst,vary2 * arg)122 vary4vescape(vary4 * dst, vary2 * arg)
123
124 {
125 int i;
126
127 for (i = 0; i < arg->len; i++)
128 if (arg->string[i] == '\'')
129 vary4nappend(dst, "'\"'\"'", 5);
130 else
131 vary4nappend(dst, arg->string + i, 1);
132 }
133
134
135 static void
vary4vargquote(vary4 * dst,vary2 * arg)136 vary4vargquote(vary4 * dst, vary2 * arg)
137
138 {
139 vary4nappend(dst, " '", 2);
140 vary4vescape(dst, arg);
141 vary4nappend(dst, "'", 1);
142 }
143
144
145 int
main(int argsc,arguments * args)146 main(int argsc, arguments * args)
147
148 {
149 vary4 cmd;
150 int i;
151 char c;
152 addelement * aelp;
153 sgmladdelement * saelp;
154
155 /* Specify additional catalogs. */
156 cmd.len = 0;
157 if (args->catalog->len) {
158 for (i = 0; i < args->catalog->len &&
159 !args->catalog->item[i].param.len; i++)
160 ;
161
162 vary4append(&cmd, "XML_CATALOG_FILES=");
163 if (i < args->catalog->len) {
164 c = '\'';
165 for (i = 0; i < args->catalog->len; i++) {
166 if (!args->catalog->item[i].param.len)
167 continue;
168 vary4nappend(&cmd, &c, 1);
169 c = ' ';
170 vary4vescape(&cmd,
171 &args->catalog->item[i].param);
172 }
173 vary4nappend(&cmd, "'", 1);
174 }
175 vary4nappend(&cmd, " ", 1);
176 }
177
178 /* find length of library name. */
179 for (i = 0; i < 10 && args->pgm[i] && args->pgm[i] != '/'; i++)
180 ;
181
182 /* Store program name in command buffer. */
183 vary4append(&cmd, "/QSYS.LIB/");
184 vary4nappend(&cmd, args->pgm, i);
185 vary4append(&cmd, ".LIB/XMLCATALOG.PGM");
186
187 /* Map command arguments to standard xmlcatalog argument vector. */
188 if (args->kind && args->kind->len)
189 vary4varg(&cmd, args->kind);
190
191 if (args->verbose && args->verbose->len)
192 vary4varg(&cmd, args->verbose);
193
194 if (args->delete)
195 for (i = 0; i < args->delete->len; i++) {
196 vary4arg(&cmd, "--del");
197 vary4vargquote(&cmd, &args->delete->item[i].param);
198 }
199
200 if (args->kind && args->kind->len) {
201 /* Process SGML-specific parameters. */
202 if (args->superupd && args->superupd->len)
203 vary4varg(&cmd, args->superupd);
204
205 if (args->sgmladd)
206 for (i = 0; i < args->sgmladd->len; i++) {
207 saelp = OFFSETBY(sgmladdelement, args->sgmladd,
208 args->sgmladd->item[i].param);
209 if (!((vary2 *) &saelp->catalog)->len)
210 continue;
211 vary4arg(&cmd, "--add");
212 vary4vargquote(&cmd, (vary2 *) &saelp->catalog);
213 vary4vargquote(&cmd, (vary2 *) &saelp->ident);
214 }
215 }
216 else {
217 /* Process XML-specific parameters. */
218 if (args->convert && args->convert->len)
219 vary4varg(&cmd, args->convert);
220
221 if (args->add)
222 for (i = 0; i < args->add->len; i++) {
223 aelp = OFFSETBY(addelement, args->add,
224 args->add->item[i].param);
225 if (!((vary2 *) &aelp->origin)->len)
226 continue;
227 vary4arg(&cmd, "--add");
228 vary4varg(&cmd, (vary2 *) &aelp->type);
229 vary4vargquote(&cmd, (vary2 *) &aelp->origin);
230 vary4vargquote(&cmd, (vary2 *) &aelp->replace);
231 }
232 }
233
234 /* Avoid INSTMF(*NEW) and OUTSMTF(*INSTMF). */
235 if (args->outstmf && args->outstmf->len && !args->outstmf->string[0])
236 if (args->instmf && args->instmf->len)
237 args->outstmf = args->instmf;
238 else
239 args->outstmf = NULL;
240
241 /* If INSTMF(*NEW) and OUTSTMF(somepath), Use --create --noout and
242 somepath as (unexisting) input file. */
243 if (args->outstmf && args->outstmf->len)
244 if (!args->instmf || !args->instmf->len) {
245 vary4arg(&cmd, "--create");
246 vary4arg(&cmd, "--noout");
247 args->instmf = args->outstmf;
248 args->outstmf = NULL;
249 }
250
251 /* If output to input file, use --noout option. */
252 if (args->instmf && args->outstmf && args->instmf->len &&
253 args->instmf->len == args->outstmf->len &&
254 !strncmp(args->instmf->string, args->outstmf->string,
255 args->instmf->len)) {
256 vary4arg(&cmd, "--noout");
257 args->outstmf = NULL;
258 }
259
260 /* If no input file create catalog, else specify the input file name. */
261 /* Specify the input file name: my be a dummy one. */
262 if (!args->instmf || !args->instmf->len) {
263 vary4arg(&cmd, "--create -");
264 vary4arg(&cmd, ".dmyxmlcatalog");
265 }
266 else {
267 vary4arg(&cmd, "-");
268 vary4vargquote(&cmd, args->instmf);
269 }
270
271 /* Query entities. */
272
273 if (args->resolve)
274 for (i = 0; i < args->resolve->len; i++)
275 vary4vargquote(&cmd, &args->resolve->item[i].param);
276
277 /* Redirect output if requested. */
278 if (args->outstmf && args->outstmf->len) {
279 vary4arg(&cmd, ">");
280 vary4vargquote(&cmd, args->outstmf);
281 }
282
283 /* Execute the shell command. */
284 qshell(&cmd);
285
286 /* Terminate. */
287 exit(0);
288 }
289