1 /**
2 *** XMLLINT 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 union { \
34 char _pad[itemsize]; \
35 itemtype param; \
36 } item[1]; \
37 }
38
39
40 /* Arguments from CL command. */
41 typedef struct {
42 char * pgm; /* Program name. */
43 vary2 * stmf; /* XML file name or URL. */
44 vary2 * dtd; /* DTD location or public identifier. */
45 char * dtdvalid; /* *DTDURL or *DTDFPI. */
46 vary2 * schema; /* Schema file name or URL. */
47 vary2 * schemakind; /* --schema/--relaxng/--schematron. */
48 vary2 * outstmf; /* Output stream file name. */
49 vary2 * xpath; /* XPath filter. */
50 vary2 * pattern; /* Reader filter pattern. */
51 paramlist(5000 + 2, vary2) * path; /* Path for resources. */
52 vary2 * pretty; /* Pretty-print style. */
53 unsigned long * maxmem; /* Maximum dynamic memory. */
54 vary2 * encoding; /* Output encoding. */
55 paramlist(20 + 2, vary2) * options; /* Other options. */
56 } arguments;
57
58
59 /* Definition of QSHELL program. */
60 extern void qshell(vary4 * cmd);
61 #pragma linkage(qshell, OS)
62 #pragma map(qshell, "QSHELL/QZSHQSHC")
63
64
65 static void
vary4nappend(vary4 * dst,const char * src,size_t len)66 vary4nappend(vary4 * dst, const char * src, size_t len)
67
68 {
69 if (len > sizeof(dst->string) - dst->len)
70 len = sizeof(dst->string) - dst->len;
71
72 if (len) {
73 memcpy(dst->string + dst->len, src, len);
74 dst->len += len;
75 }
76 }
77
78
79 static void
vary4append(vary4 * dst,const char * src)80 vary4append(vary4 * dst, const char * src)
81
82 {
83 vary4nappend(dst, src, strlen(src));
84 }
85
86
87 static void
vary4arg(vary4 * dst,const char * arg)88 vary4arg(vary4 * dst, const char * arg)
89
90 {
91 vary4nappend(dst, " ", 1);
92 vary4append(dst, arg);
93 }
94
95
96 static void
vary4varg(vary4 * dst,vary2 * arg)97 vary4varg(vary4 * dst, vary2 * arg)
98
99 {
100 vary4nappend(dst, " ", 1);
101 vary4nappend(dst, arg->string, arg->len);
102 }
103
104
105 static void
vary4vescape(vary4 * dst,vary2 * arg)106 vary4vescape(vary4 * dst, vary2 * arg)
107
108 {
109 int i;
110
111 for (i = 0; i < arg->len; i++)
112 if (arg->string[i] == '\'')
113 vary4nappend(dst, "'\"'\"'", 5);
114 else
115 vary4nappend(dst, arg->string + i, 1);
116 }
117
118
119 static void
vary4vargquote(vary4 * dst,vary2 * arg)120 vary4vargquote(vary4 * dst, vary2 * arg)
121
122 {
123 vary4nappend(dst, " '", 2);
124 vary4vescape(dst, arg);
125 vary4nappend(dst, "'", 1);
126 }
127
128
129 int
main(int argsc,arguments * args)130 main(int argsc, arguments * args)
131
132 {
133 vary4 cmd;
134 int i;
135 char textbuf[20];
136 char * lang;
137
138 /* find length of library name. */
139 for (i = 0; i < 10 && args->pgm[i] && args->pgm[i] != '/'; i++)
140 ;
141
142 /* Store program name in command buffer. */
143 cmd.len = 0;
144 vary4append(&cmd, "/QSYS.LIB/");
145 vary4nappend(&cmd, args->pgm, i);
146 vary4append(&cmd, ".LIB/XMLLINT.PGM");
147
148 /* Map command arguments to standard xmllint argument vector. */
149
150 if (args->dtd && args->dtd->len) {
151 if (args->dtdvalid && args->dtdvalid[4] == 'F')
152 vary4arg(&cmd, "--dtdvalidfpi");
153 else
154 vary4arg(&cmd, "--dtdvalid");
155
156 vary4vargquote(&cmd, args->dtd);
157 }
158
159 if (args->schema && args->schema->len) {
160 vary4varg(&cmd, args->schemakind);
161 vary4vargquote(&cmd, args->schema);
162 }
163
164 if (args->outstmf && args->outstmf->len) {
165 vary4arg(&cmd, "--output");
166 vary4vargquote(&cmd, args->outstmf);
167
168 if (args->encoding && args->encoding->len) {
169 vary4arg(&cmd, "--encoding");
170 vary4vargquote(&cmd, args->encoding);
171 }
172 }
173
174 if (args->xpath && args->xpath->len) {
175 vary4arg(&cmd, "--xpath");
176 vary4vargquote(&cmd, args->xpath);
177 }
178
179 if (args->pattern && args->pattern->len) {
180 vary4arg(&cmd, "--pattern");
181 vary4vargquote(&cmd, args->pattern);
182 }
183
184 if (args->path && args->path->len) {
185 vary4arg(&cmd, "--path '");
186 vary4vescape(&cmd, &args->path->item[0].param);
187 for (i = 1; i < args->path->len; i++) {
188 vary4nappend(&cmd, ":", 1);
189 vary4vescape(&cmd, &args->path->item[i].param);
190 }
191 vary4nappend(&cmd, "'", 1);
192 }
193
194 if (args->pretty && args->pretty->len &&
195 args->pretty->string[0] != '0') {
196 vary4arg(&cmd, "--pretty");
197 vary4varg(&cmd, args->pretty);
198 }
199
200 if (args->maxmem && *args->maxmem) {
201 snprintf(textbuf, sizeof textbuf, "%lu", *args->maxmem);
202 vary4arg(&cmd, "--maxmem");
203 vary4arg(&cmd, textbuf);
204 }
205
206 for (i = 0; i < args->options->len; i++)
207 vary4varg(&cmd, &args->options->item[i].param);
208
209 vary4vargquote(&cmd, args->stmf);
210
211 /* Execute the shell command. */
212 qshell(&cmd);
213
214 /* Terminate. */
215 exit(0);
216 }
217