1*6236dae4SAndroid Build Coastguard Worker /***************************************************************************
2*6236dae4SAndroid Build Coastguard Worker * _ _ ____ _
3*6236dae4SAndroid Build Coastguard Worker * Project ___| | | | _ \| |
4*6236dae4SAndroid Build Coastguard Worker * / __| | | | |_) | |
5*6236dae4SAndroid Build Coastguard Worker * | (__| |_| | _ <| |___
6*6236dae4SAndroid Build Coastguard Worker * \___|\___/|_| \_\_____|
7*6236dae4SAndroid Build Coastguard Worker *
8*6236dae4SAndroid Build Coastguard Worker * Copyright (C) Daniel Stenberg, <[email protected]>, et al.
9*6236dae4SAndroid Build Coastguard Worker *
10*6236dae4SAndroid Build Coastguard Worker * This software is licensed as described in the file COPYING, which
11*6236dae4SAndroid Build Coastguard Worker * you should have received as part of this distribution. The terms
12*6236dae4SAndroid Build Coastguard Worker * are also available at https://curl.se/docs/copyright.html.
13*6236dae4SAndroid Build Coastguard Worker *
14*6236dae4SAndroid Build Coastguard Worker * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15*6236dae4SAndroid Build Coastguard Worker * copies of the Software, and permit persons to whom the Software is
16*6236dae4SAndroid Build Coastguard Worker * furnished to do so, under the terms of the COPYING file.
17*6236dae4SAndroid Build Coastguard Worker *
18*6236dae4SAndroid Build Coastguard Worker * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19*6236dae4SAndroid Build Coastguard Worker * KIND, either express or implied.
20*6236dae4SAndroid Build Coastguard Worker *
21*6236dae4SAndroid Build Coastguard Worker * SPDX-License-Identifier: curl
22*6236dae4SAndroid Build Coastguard Worker *
23*6236dae4SAndroid Build Coastguard Worker ***************************************************************************/
24*6236dae4SAndroid Build Coastguard Worker #include "tool_setup.h"
25*6236dae4SAndroid Build Coastguard Worker
26*6236dae4SAndroid Build Coastguard Worker #ifdef __VMS
27*6236dae4SAndroid Build Coastguard Worker
28*6236dae4SAndroid Build Coastguard Worker #if defined(__DECC) && !defined(__VAX) && \
29*6236dae4SAndroid Build Coastguard Worker defined(__CRTL_VER) && (__CRTL_VER >= 70301000)
30*6236dae4SAndroid Build Coastguard Worker #include <unixlib.h>
31*6236dae4SAndroid Build Coastguard Worker #endif
32*6236dae4SAndroid Build Coastguard Worker
33*6236dae4SAndroid Build Coastguard Worker #include "curlx.h"
34*6236dae4SAndroid Build Coastguard Worker
35*6236dae4SAndroid Build Coastguard Worker #include "curlmsg_vms.h"
36*6236dae4SAndroid Build Coastguard Worker #include "tool_vms.h"
37*6236dae4SAndroid Build Coastguard Worker
38*6236dae4SAndroid Build Coastguard Worker #include "memdebug.h" /* keep this as LAST include */
39*6236dae4SAndroid Build Coastguard Worker
40*6236dae4SAndroid Build Coastguard Worker void decc$__posix_exit(int __status);
41*6236dae4SAndroid Build Coastguard Worker void decc$exit(int __status);
42*6236dae4SAndroid Build Coastguard Worker
43*6236dae4SAndroid Build Coastguard Worker static int vms_shell = -1;
44*6236dae4SAndroid Build Coastguard Worker
45*6236dae4SAndroid Build Coastguard Worker /* VMS has a DCL shell and also has Unix shells ported to it.
46*6236dae4SAndroid Build Coastguard Worker * When curl is running under a Unix shell, we want it to be as much
47*6236dae4SAndroid Build Coastguard Worker * like Unix as possible.
48*6236dae4SAndroid Build Coastguard Worker */
is_vms_shell(void)49*6236dae4SAndroid Build Coastguard Worker int is_vms_shell(void)
50*6236dae4SAndroid Build Coastguard Worker {
51*6236dae4SAndroid Build Coastguard Worker char *shell;
52*6236dae4SAndroid Build Coastguard Worker
53*6236dae4SAndroid Build Coastguard Worker /* Have we checked the shell yet? */
54*6236dae4SAndroid Build Coastguard Worker if(vms_shell >= 0)
55*6236dae4SAndroid Build Coastguard Worker return vms_shell;
56*6236dae4SAndroid Build Coastguard Worker
57*6236dae4SAndroid Build Coastguard Worker shell = getenv("SHELL");
58*6236dae4SAndroid Build Coastguard Worker
59*6236dae4SAndroid Build Coastguard Worker /* No shell, means DCL */
60*6236dae4SAndroid Build Coastguard Worker if(!shell) {
61*6236dae4SAndroid Build Coastguard Worker vms_shell = 1;
62*6236dae4SAndroid Build Coastguard Worker return 1;
63*6236dae4SAndroid Build Coastguard Worker }
64*6236dae4SAndroid Build Coastguard Worker
65*6236dae4SAndroid Build Coastguard Worker /* Have to make sure some one did not set shell to DCL */
66*6236dae4SAndroid Build Coastguard Worker if(strcmp(shell, "DCL") == 0) {
67*6236dae4SAndroid Build Coastguard Worker vms_shell = 1;
68*6236dae4SAndroid Build Coastguard Worker return 1;
69*6236dae4SAndroid Build Coastguard Worker }
70*6236dae4SAndroid Build Coastguard Worker
71*6236dae4SAndroid Build Coastguard Worker vms_shell = 0;
72*6236dae4SAndroid Build Coastguard Worker return 0;
73*6236dae4SAndroid Build Coastguard Worker }
74*6236dae4SAndroid Build Coastguard Worker
75*6236dae4SAndroid Build Coastguard Worker /*
76*6236dae4SAndroid Build Coastguard Worker * VMS has two exit() routines. When running under a Unix style shell, then
77*6236dae4SAndroid Build Coastguard Worker * Unix style and the __posix_exit() routine is used.
78*6236dae4SAndroid Build Coastguard Worker *
79*6236dae4SAndroid Build Coastguard Worker * When running under the DCL shell, then the VMS encoded codes and decc$exit()
80*6236dae4SAndroid Build Coastguard Worker * is used.
81*6236dae4SAndroid Build Coastguard Worker *
82*6236dae4SAndroid Build Coastguard Worker * We can not use exit() or return a code from main() because the actual
83*6236dae4SAndroid Build Coastguard Worker * routine called depends on both the compiler version, compile options, and
84*6236dae4SAndroid Build Coastguard Worker * feature macro settings, and one of the exit routines is hidden at compile
85*6236dae4SAndroid Build Coastguard Worker * time.
86*6236dae4SAndroid Build Coastguard Worker *
87*6236dae4SAndroid Build Coastguard Worker * Since we want Curl to work properly under the VMS DCL shell and Unix
88*6236dae4SAndroid Build Coastguard Worker * shells under VMS, this routine should compile correctly regardless of
89*6236dae4SAndroid Build Coastguard Worker * the settings.
90*6236dae4SAndroid Build Coastguard Worker */
91*6236dae4SAndroid Build Coastguard Worker
vms_special_exit(int code,int vms_show)92*6236dae4SAndroid Build Coastguard Worker void vms_special_exit(int code, int vms_show)
93*6236dae4SAndroid Build Coastguard Worker {
94*6236dae4SAndroid Build Coastguard Worker int vms_code;
95*6236dae4SAndroid Build Coastguard Worker
96*6236dae4SAndroid Build Coastguard Worker /* The POSIX exit mode is only available after VMS 7.0 */
97*6236dae4SAndroid Build Coastguard Worker #if __CRTL_VER >= 70000000
98*6236dae4SAndroid Build Coastguard Worker if(is_vms_shell() == 0) {
99*6236dae4SAndroid Build Coastguard Worker decc$__posix_exit(code);
100*6236dae4SAndroid Build Coastguard Worker }
101*6236dae4SAndroid Build Coastguard Worker #endif
102*6236dae4SAndroid Build Coastguard Worker
103*6236dae4SAndroid Build Coastguard Worker if(code > CURL_LAST) { /* If CURL_LAST exceeded then */
104*6236dae4SAndroid Build Coastguard Worker vms_code = CURL_LAST; /* curlmsg.h is out of sync. */
105*6236dae4SAndroid Build Coastguard Worker }
106*6236dae4SAndroid Build Coastguard Worker else {
107*6236dae4SAndroid Build Coastguard Worker vms_code = vms_cond[code] | vms_show;
108*6236dae4SAndroid Build Coastguard Worker }
109*6236dae4SAndroid Build Coastguard Worker decc$exit(vms_code);
110*6236dae4SAndroid Build Coastguard Worker }
111*6236dae4SAndroid Build Coastguard Worker
112*6236dae4SAndroid Build Coastguard Worker #if defined(__DECC) && !defined(__VAX) && \
113*6236dae4SAndroid Build Coastguard Worker defined(__CRTL_VER) && (__CRTL_VER >= 70301000)
114*6236dae4SAndroid Build Coastguard Worker
115*6236dae4SAndroid Build Coastguard Worker /*
116*6236dae4SAndroid Build Coastguard Worker * 2004-09-19 SMS.
117*6236dae4SAndroid Build Coastguard Worker *
118*6236dae4SAndroid Build Coastguard Worker * decc_init()
119*6236dae4SAndroid Build Coastguard Worker *
120*6236dae4SAndroid Build Coastguard Worker * On non-VAX systems, use LIB$INITIALIZE to set a collection of C
121*6236dae4SAndroid Build Coastguard Worker * RTL features without using the DECC$* logical name method, nor
122*6236dae4SAndroid Build Coastguard Worker * requiring the user to define the corresponding logical names.
123*6236dae4SAndroid Build Coastguard Worker */
124*6236dae4SAndroid Build Coastguard Worker
125*6236dae4SAndroid Build Coastguard Worker /* Structure to hold a DECC$* feature name and its desired value. */
126*6236dae4SAndroid Build Coastguard Worker struct decc_feat_t {
127*6236dae4SAndroid Build Coastguard Worker char *name;
128*6236dae4SAndroid Build Coastguard Worker int value;
129*6236dae4SAndroid Build Coastguard Worker };
130*6236dae4SAndroid Build Coastguard Worker
131*6236dae4SAndroid Build Coastguard Worker /* Array of DECC$* feature names and their desired values. */
132*6236dae4SAndroid Build Coastguard Worker static const struct decc_feat_t decc_feat_array[] = {
133*6236dae4SAndroid Build Coastguard Worker /* Preserve command-line case with SET PROCESS/PARSE_STYLE=EXTENDED */
134*6236dae4SAndroid Build Coastguard Worker { "DECC$ARGV_PARSE_STYLE", 1 },
135*6236dae4SAndroid Build Coastguard Worker /* Preserve case for filenames on ODS5 disks. */
136*6236dae4SAndroid Build Coastguard Worker { "DECC$EFS_CASE_PRESERVE", 1 },
137*6236dae4SAndroid Build Coastguard Worker /* Enable multiple dots (and most characters) in ODS5 filenames,
138*6236dae4SAndroid Build Coastguard Worker while preserving VMS-ness of ";version". */
139*6236dae4SAndroid Build Coastguard Worker { "DECC$EFS_CHARSET", 1 },
140*6236dae4SAndroid Build Coastguard Worker /* List terminator. */
141*6236dae4SAndroid Build Coastguard Worker { (char *)NULL, 0 }
142*6236dae4SAndroid Build Coastguard Worker };
143*6236dae4SAndroid Build Coastguard Worker
144*6236dae4SAndroid Build Coastguard Worker /* Flag to sense if decc_init() was called. */
145*6236dae4SAndroid Build Coastguard Worker static int decc_init_done = -1;
146*6236dae4SAndroid Build Coastguard Worker
147*6236dae4SAndroid Build Coastguard Worker /* LIB$INITIALIZE initialization function. */
decc_init(void)148*6236dae4SAndroid Build Coastguard Worker static void decc_init(void)
149*6236dae4SAndroid Build Coastguard Worker {
150*6236dae4SAndroid Build Coastguard Worker int feat_index;
151*6236dae4SAndroid Build Coastguard Worker int feat_value;
152*6236dae4SAndroid Build Coastguard Worker int feat_value_max;
153*6236dae4SAndroid Build Coastguard Worker int feat_value_min;
154*6236dae4SAndroid Build Coastguard Worker int i;
155*6236dae4SAndroid Build Coastguard Worker int sts;
156*6236dae4SAndroid Build Coastguard Worker
157*6236dae4SAndroid Build Coastguard Worker /* Set the global flag to indicate that LIB$INITIALIZE worked. */
158*6236dae4SAndroid Build Coastguard Worker decc_init_done = 1;
159*6236dae4SAndroid Build Coastguard Worker
160*6236dae4SAndroid Build Coastguard Worker /* Loop through all items in the decc_feat_array[]. */
161*6236dae4SAndroid Build Coastguard Worker for(i = 0; decc_feat_array[i].name != NULL; i++) {
162*6236dae4SAndroid Build Coastguard Worker
163*6236dae4SAndroid Build Coastguard Worker /* Get the feature index. */
164*6236dae4SAndroid Build Coastguard Worker feat_index = decc$feature_get_index(decc_feat_array[i].name);
165*6236dae4SAndroid Build Coastguard Worker
166*6236dae4SAndroid Build Coastguard Worker if(feat_index >= 0) {
167*6236dae4SAndroid Build Coastguard Worker /* Valid item. Collect its properties. */
168*6236dae4SAndroid Build Coastguard Worker feat_value = decc$feature_get_value(feat_index, 1);
169*6236dae4SAndroid Build Coastguard Worker feat_value_min = decc$feature_get_value(feat_index, 2);
170*6236dae4SAndroid Build Coastguard Worker feat_value_max = decc$feature_get_value(feat_index, 3);
171*6236dae4SAndroid Build Coastguard Worker
172*6236dae4SAndroid Build Coastguard Worker if((decc_feat_array[i].value >= feat_value_min) &&
173*6236dae4SAndroid Build Coastguard Worker (decc_feat_array[i].value <= feat_value_max)) {
174*6236dae4SAndroid Build Coastguard Worker /* Valid value. Set it if necessary. */
175*6236dae4SAndroid Build Coastguard Worker if(feat_value != decc_feat_array[i].value) {
176*6236dae4SAndroid Build Coastguard Worker sts = decc$feature_set_value(feat_index, 1,
177*6236dae4SAndroid Build Coastguard Worker decc_feat_array[i].value);
178*6236dae4SAndroid Build Coastguard Worker }
179*6236dae4SAndroid Build Coastguard Worker }
180*6236dae4SAndroid Build Coastguard Worker else {
181*6236dae4SAndroid Build Coastguard Worker /* Invalid DECC feature value. */
182*6236dae4SAndroid Build Coastguard Worker printf(" INVALID DECC FEATURE VALUE, %d: %d <= %s <= %d.\n",
183*6236dae4SAndroid Build Coastguard Worker feat_value,
184*6236dae4SAndroid Build Coastguard Worker feat_value_min, decc_feat_array[i].name, feat_value_max);
185*6236dae4SAndroid Build Coastguard Worker }
186*6236dae4SAndroid Build Coastguard Worker }
187*6236dae4SAndroid Build Coastguard Worker else {
188*6236dae4SAndroid Build Coastguard Worker /* Invalid DECC feature name. */
189*6236dae4SAndroid Build Coastguard Worker printf(" UNKNOWN DECC FEATURE: %s.\n", decc_feat_array[i].name);
190*6236dae4SAndroid Build Coastguard Worker }
191*6236dae4SAndroid Build Coastguard Worker
192*6236dae4SAndroid Build Coastguard Worker }
193*6236dae4SAndroid Build Coastguard Worker }
194*6236dae4SAndroid Build Coastguard Worker
195*6236dae4SAndroid Build Coastguard Worker /* Get "decc_init()" into a valid, loaded LIB$INITIALIZE PSECT. */
196*6236dae4SAndroid Build Coastguard Worker
197*6236dae4SAndroid Build Coastguard Worker #pragma nostandard
198*6236dae4SAndroid Build Coastguard Worker
199*6236dae4SAndroid Build Coastguard Worker /* Establish the LIB$INITIALIZE PSECTs, with proper alignment and
200*6236dae4SAndroid Build Coastguard Worker other attributes. Note that "nopic" is significant only on VAX. */
201*6236dae4SAndroid Build Coastguard Worker #pragma extern_model save
202*6236dae4SAndroid Build Coastguard Worker #pragma extern_model strict_refdef "LIB$INITIALIZ" 2, nopic, nowrt
203*6236dae4SAndroid Build Coastguard Worker const int spare[8] = {0};
204*6236dae4SAndroid Build Coastguard Worker #pragma extern_model strict_refdef "LIB$INITIALIZE" 2, nopic, nowrt
205*6236dae4SAndroid Build Coastguard Worker void (*const x_decc_init)() = decc_init;
206*6236dae4SAndroid Build Coastguard Worker #pragma extern_model restore
207*6236dae4SAndroid Build Coastguard Worker
208*6236dae4SAndroid Build Coastguard Worker /* Fake reference to ensure loading the LIB$INITIALIZE PSECT. */
209*6236dae4SAndroid Build Coastguard Worker #pragma extern_model save
210*6236dae4SAndroid Build Coastguard Worker int LIB$INITIALIZE(void);
211*6236dae4SAndroid Build Coastguard Worker #pragma extern_model strict_refdef
212*6236dae4SAndroid Build Coastguard Worker int dmy_lib$initialize = (int) LIB$INITIALIZE;
213*6236dae4SAndroid Build Coastguard Worker #pragma extern_model restore
214*6236dae4SAndroid Build Coastguard Worker
215*6236dae4SAndroid Build Coastguard Worker #pragma standard
216*6236dae4SAndroid Build Coastguard Worker
217*6236dae4SAndroid Build Coastguard Worker #endif /* __DECC && !__VAX && __CRTL_VER && __CRTL_VER >= 70301000 */
218*6236dae4SAndroid Build Coastguard Worker
219*6236dae4SAndroid Build Coastguard Worker #endif /* __VMS */
220