xref: /plan9/sys/src/cmd/troff/dwbinit.c (revision 219b2ee8daee37f4aad58d63f21287faa8e4ffdc)
1*219b2ee8SDavid du Colombier /*
2*219b2ee8SDavid du Colombier  *
3*219b2ee8SDavid du Colombier  * Pathname management routines for DWB C programs.
4*219b2ee8SDavid du Colombier  *
5*219b2ee8SDavid du Colombier  * Applications should initialize a dwbinit array with the string
6*219b2ee8SDavid du Colombier  * pointers and arrays that need to be updated, and then hand that
7*219b2ee8SDavid du Colombier  * array to DWBinit before much else happens in their main program.
8*219b2ee8SDavid du Colombier  * DWBinit calls DWBhome to get the current home directory. DWBhome
9*219b2ee8SDavid du Colombier  * uses the last definition of DWBENV (usually "DWBHOME") in file
10*219b2ee8SDavid du Colombier  * DWBCONFIG (e.g., /usr/lib/dwb3.4) or the value assigned to that
11*219b2ee8SDavid du Colombier  * variable in the environment if the DWBCONFIG file doesn't exist,
12*219b2ee8SDavid du Colombier  * can't be read, or doesn't define DWBENV.
13*219b2ee8SDavid du Colombier  *
14*219b2ee8SDavid du Colombier  * DWBCONFIG must be a simple shell script - comments, a definition
15*219b2ee8SDavid du Colombier  * of DWBHOME, and perhaps an export or echo is about all that's
16*219b2ee8SDavid du Colombier  * allowed. The parsing in DWBhome is simple and makes no attempt
17*219b2ee8SDavid du Colombier  * to duplicate the shell. It only looks for DWBHOME= as the first
18*219b2ee8SDavid du Colombier  * non-white space string on a line, so
19*219b2ee8SDavid du Colombier  *
20*219b2ee8SDavid du Colombier  *	#
21*219b2ee8SDavid du Colombier  *	# A sample DWBCONFIG shell script
22*219b2ee8SDavid du Colombier  *	#
23*219b2ee8SDavid du Colombier  *
24*219b2ee8SDavid du Colombier  *	DWBHOME=/usr/add-on/dwb3.4
25*219b2ee8SDavid du Colombier  *	export DWBHOME
26*219b2ee8SDavid du Colombier  *
27*219b2ee8SDavid du Colombier  * means DWBhome would return "/usr/add-on/dwb3.4" for the DWB home
28*219b2ee8SDavid du Colombier  * directory. A DWBCONFIG file means there can only be one working
29*219b2ee8SDavid du Colombier  * copy of a DWB release on a system, which seems like a good idea.
30*219b2ee8SDavid du Colombier  * Using DWBCONFIG also means programs will always include correct
31*219b2ee8SDavid du Colombier  * versions of files (e.g., prologues or macro packages).
32*219b2ee8SDavid du Colombier  *
33*219b2ee8SDavid du Colombier  * Relying on an environment variable guarantees nothing. You could
34*219b2ee8SDavid du Colombier  * execute a version of dpost, but your environment might point at
35*219b2ee8SDavid du Colombier  * incorrect font tables or prologues. Despite the obvious problems
36*219b2ee8SDavid du Colombier  * we've also implemented an environment variable approach, but it's
37*219b2ee8SDavid du Colombier  * only used if there's no DWBCONFIG file.
38*219b2ee8SDavid du Colombier  *
39*219b2ee8SDavid du Colombier  * DWBinit calls DWBhome to get the DWB home directory prefix and
40*219b2ee8SDavid du Colombier  * then marches through its dwbinit argument, removing the default
41*219b2ee8SDavid du Colombier  * home directory and prepending the new home. DWBinit stops when
42*219b2ee8SDavid du Colombier  * it reaches an element that has NULL for its address and value
43*219b2ee8SDavid du Colombier  * fields. Pointers in a dwbinit array are reallocated and properly
44*219b2ee8SDavid du Colombier  * initialized; arrays are simply reinitialized if there's room.
45*219b2ee8SDavid du Colombier  * All pathnames that are to be adjusted should be relative. For
46*219b2ee8SDavid du Colombier  * example,
47*219b2ee8SDavid du Colombier  *
48*219b2ee8SDavid du Colombier  *	char	*fontdir = "lib/font";
49*219b2ee8SDavid du Colombier  *	char	xyzzy[25] = "etc/xyzzy";
50*219b2ee8SDavid du Colombier  *
51*219b2ee8SDavid du Colombier  * would be represented in a dwbinit array as,
52*219b2ee8SDavid du Colombier  *
53*219b2ee8SDavid du Colombier  *	dwbinit allpaths[] = {
54*219b2ee8SDavid du Colombier  *		&fontdir, NULL, 0,
55*219b2ee8SDavid du Colombier  *		NULL, xyzzy, sizeof(xyzzy),
56*219b2ee8SDavid du Colombier  *		NULL, NULL, 0
57*219b2ee8SDavid du Colombier  *	};
58*219b2ee8SDavid du Colombier  *
59*219b2ee8SDavid du Colombier  * The last element must have NULL entries for the address and
60*219b2ee8SDavid du Colombier  * value fields. The main() routine would then do,
61*219b2ee8SDavid du Colombier  *
62*219b2ee8SDavid du Colombier  *	#include "dwbinit.h"
63*219b2ee8SDavid du Colombier  *
64*219b2ee8SDavid du Colombier  *	main() {
65*219b2ee8SDavid du Colombier  *
66*219b2ee8SDavid du Colombier  *		DWBinit("program name", allpaths);
67*219b2ee8SDavid du Colombier  *		...
68*219b2ee8SDavid du Colombier  *	}
69*219b2ee8SDavid du Colombier  *
70*219b2ee8SDavid du Colombier  * Debugging is enabled if DWBDEBUG is in the environment and has
71*219b2ee8SDavid du Colombier  * the value ON. Output is occasionally useful and probably should
72*219b2ee8SDavid du Colombier  * be documented.
73*219b2ee8SDavid du Colombier  *
74*219b2ee8SDavid du Colombier  */
75*219b2ee8SDavid du Colombier 
76*219b2ee8SDavid du Colombier #include <stdio.h>
77*219b2ee8SDavid du Colombier #include <ctype.h>
78*219b2ee8SDavid du Colombier #include <string.h>
79*219b2ee8SDavid du Colombier #include <stdlib.h>
80*219b2ee8SDavid du Colombier 
81*219b2ee8SDavid du Colombier #include "dwbinit.h"
82*219b2ee8SDavid du Colombier 
83*219b2ee8SDavid du Colombier #ifndef DWBCONFIG
84*219b2ee8SDavid du Colombier #define DWBCONFIG	"/dev/null"
85*219b2ee8SDavid du Colombier #endif
86*219b2ee8SDavid du Colombier 
87*219b2ee8SDavid du Colombier #ifndef DWBENV
88*219b2ee8SDavid du Colombier #define DWBENV		"DWBHOME"
89*219b2ee8SDavid du Colombier #endif
90*219b2ee8SDavid du Colombier 
91*219b2ee8SDavid du Colombier #ifndef DWBHOME
92*219b2ee8SDavid du Colombier #define DWBHOME		""
93*219b2ee8SDavid du Colombier #endif
94*219b2ee8SDavid du Colombier 
95*219b2ee8SDavid du Colombier #ifndef DWBDEBUG
96*219b2ee8SDavid du Colombier #define DWBDEBUG	"DWBDEBUG"
97*219b2ee8SDavid du Colombier #endif
98*219b2ee8SDavid du Colombier 
99*219b2ee8SDavid du Colombier #ifndef DWBPREFIX
100*219b2ee8SDavid du Colombier #define DWBPREFIX	"\\*(.P"
101*219b2ee8SDavid du Colombier #endif
102*219b2ee8SDavid du Colombier 
103*219b2ee8SDavid du Colombier /*****************************************************************************/
104*219b2ee8SDavid du Colombier 
DWBdebug(dwbinit * ptr,int level)105*219b2ee8SDavid du Colombier void DWBdebug(dwbinit *ptr, int level)
106*219b2ee8SDavid du Colombier {
107*219b2ee8SDavid du Colombier 
108*219b2ee8SDavid du Colombier     char	*path;
109*219b2ee8SDavid du Colombier     char	*home;
110*219b2ee8SDavid du Colombier     static char	*debug = NULL;
111*219b2ee8SDavid du Colombier 
112*219b2ee8SDavid du Colombier /*
113*219b2ee8SDavid du Colombier  *
114*219b2ee8SDavid du Colombier  * Debugging output, but only if DWBDEBUG is defined to be ON in the
115*219b2ee8SDavid du Colombier  * environment. Dumps general info the first time through.
116*219b2ee8SDavid du Colombier  *
117*219b2ee8SDavid du Colombier  */
118*219b2ee8SDavid du Colombier 
119*219b2ee8SDavid du Colombier     if ( debug == NULL && (debug = getenv(DWBDEBUG)) == NULL )
120*219b2ee8SDavid du Colombier 	debug = "OFF";
121*219b2ee8SDavid du Colombier 
122*219b2ee8SDavid du Colombier     if ( strcmp(debug, "ON") == 0 ) {
123*219b2ee8SDavid du Colombier 	if ( level == 0 ) {
124*219b2ee8SDavid du Colombier 	    fprintf(stderr, "Environment variable: %s\n", DWBENV);
125*219b2ee8SDavid du Colombier 	    fprintf(stderr, "Configuration file: %s\n", DWBCONFIG);
126*219b2ee8SDavid du Colombier 	    fprintf(stderr, "Default home: %s\n", DWBHOME);
127*219b2ee8SDavid du Colombier 	    if ( (home = DWBhome()) != NULL )
128*219b2ee8SDavid du Colombier 		fprintf(stderr, "Current home: %s\n", home);
129*219b2ee8SDavid du Colombier 	}   /* End if */
130*219b2ee8SDavid du Colombier 
131*219b2ee8SDavid du Colombier 	fprintf(stderr, "\n%s pathnames:\n", level == 0 ? "Original" : "Final");
132*219b2ee8SDavid du Colombier 	for ( ; ptr->value != NULL || ptr->address != NULL; ptr++ ) {
133*219b2ee8SDavid du Colombier 	    if ( (path = ptr->value) == NULL ) {
134*219b2ee8SDavid du Colombier 		path = *ptr->address;
135*219b2ee8SDavid du Colombier 		fprintf(stderr, " pointer: %s\n", path);
136*219b2ee8SDavid du Colombier 	    } else fprintf(stderr, " array[%d]: %s\n", ptr->length, path);
137*219b2ee8SDavid du Colombier 	    if ( level == 0 && *path == '/' )
138*219b2ee8SDavid du Colombier 		fprintf(stderr, "  WARNING - absolute path\n");
139*219b2ee8SDavid du Colombier 	}   /* End for */
140*219b2ee8SDavid du Colombier     }	/* End if */
141*219b2ee8SDavid du Colombier 
142*219b2ee8SDavid du Colombier }   /* End of DWBdebug */
143*219b2ee8SDavid du Colombier 
144*219b2ee8SDavid du Colombier /*****************************************************************************/
145*219b2ee8SDavid du Colombier 
DWBhome(void)146*219b2ee8SDavid du Colombier char *DWBhome(void)
147*219b2ee8SDavid du Colombier {
148*219b2ee8SDavid du Colombier 
149*219b2ee8SDavid du Colombier     FILE	*fp;
150*219b2ee8SDavid du Colombier     char	*ptr;
151*219b2ee8SDavid du Colombier     char	*path;
152*219b2ee8SDavid du Colombier     int		len;
153*219b2ee8SDavid du Colombier     char	buf[200];
154*219b2ee8SDavid du Colombier     char	*home = NULL;
155*219b2ee8SDavid du Colombier 
156*219b2ee8SDavid du Colombier /*
157*219b2ee8SDavid du Colombier  *
158*219b2ee8SDavid du Colombier  * Return the DWB home directory. Uses the last definition of DWBENV
159*219b2ee8SDavid du Colombier  * (usually "DWBHOME") in file DWBCONFIG (perhaps /usr/lib/dwb3.4) or
160*219b2ee8SDavid du Colombier  * the value assigned to the variable named by the DWBENV string in
161*219b2ee8SDavid du Colombier  * the environment if DWBCONFIG doesn't exist or doesn't define DWBENV.
162*219b2ee8SDavid du Colombier  * Skips the file lookup if DWBCONFIG can't be read. Returns NULL if
163*219b2ee8SDavid du Colombier  * there's no home directory.
164*219b2ee8SDavid du Colombier  *
165*219b2ee8SDavid du Colombier  */
166*219b2ee8SDavid du Colombier 
167*219b2ee8SDavid du Colombier     if ( (fp = fopen(DWBCONFIG, "r")) != NULL ) {
168*219b2ee8SDavid du Colombier 	len = strlen(DWBENV);
169*219b2ee8SDavid du Colombier 	while ( fgets(buf, sizeof(buf), fp) != NULL ) {
170*219b2ee8SDavid du Colombier 	    for ( ptr = buf; isspace(*ptr); ptr++ ) ;
171*219b2ee8SDavid du Colombier 	    if ( strncmp(ptr, DWBENV, len) == 0 && *(ptr+len) == '=' ) {
172*219b2ee8SDavid du Colombier 		path = ptr + len + 1;
173*219b2ee8SDavid du Colombier 		for ( ptr = path; !isspace(*ptr) && *ptr != ';'; ptr++ ) ;
174*219b2ee8SDavid du Colombier 		*ptr = '\0';
175*219b2ee8SDavid du Colombier 		if ( home != NULL )
176*219b2ee8SDavid du Colombier 		    free(home);
177*219b2ee8SDavid du Colombier 		if ( (home = malloc(strlen(path)+1)) != NULL )
178*219b2ee8SDavid du Colombier 		    strcpy(home, path);
179*219b2ee8SDavid du Colombier 	    }	/* End if */
180*219b2ee8SDavid du Colombier 	}   /* End while */
181*219b2ee8SDavid du Colombier 	fclose(fp);
182*219b2ee8SDavid du Colombier     }   /* End if */
183*219b2ee8SDavid du Colombier 
184*219b2ee8SDavid du Colombier     if ( home == NULL ) {
185*219b2ee8SDavid du Colombier 	if ( (home = getenv(DWBENV)) == NULL ) {
186*219b2ee8SDavid du Colombier 	    if ( (home = DWBHOME) == NULL || *home == '\0' || *home == ' ' )
187*219b2ee8SDavid du Colombier 		home = NULL;
188*219b2ee8SDavid du Colombier 	}   /* End if */
189*219b2ee8SDavid du Colombier     }	/* End if */
190*219b2ee8SDavid du Colombier 
191*219b2ee8SDavid du Colombier     while (home && *home == '/' && *(home +1) == '/')	/* remove extra slashes */
192*219b2ee8SDavid du Colombier 	home++;
193*219b2ee8SDavid du Colombier     return(home);
194*219b2ee8SDavid du Colombier 
195*219b2ee8SDavid du Colombier }   /* End of DWBhome */
196*219b2ee8SDavid du Colombier 
197*219b2ee8SDavid du Colombier /*****************************************************************************/
198*219b2ee8SDavid du Colombier 
DWBinit(char * prog,dwbinit * paths)199*219b2ee8SDavid du Colombier void DWBinit(char *prog, dwbinit *paths)
200*219b2ee8SDavid du Colombier {
201*219b2ee8SDavid du Colombier 
202*219b2ee8SDavid du Colombier     char	*prefix;
203*219b2ee8SDavid du Colombier     char	*value;
204*219b2ee8SDavid du Colombier     char	*path;
205*219b2ee8SDavid du Colombier     int		plen;
206*219b2ee8SDavid du Colombier     int		length;
207*219b2ee8SDavid du Colombier     dwbinit	*opaths = paths;
208*219b2ee8SDavid du Colombier 
209*219b2ee8SDavid du Colombier /*
210*219b2ee8SDavid du Colombier  *
211*219b2ee8SDavid du Colombier  * Adjust the pathnames listed in paths, using the home directory
212*219b2ee8SDavid du Colombier  * returned by DWBhome(). Stops when it reaches an element that has
213*219b2ee8SDavid du Colombier  * NULL address and value fields. Assumes pathnames are relative,
214*219b2ee8SDavid du Colombier  * but changes everything. DWBdebug issues a warning if an original
215*219b2ee8SDavid du Colombier  * path begins with a /.
216*219b2ee8SDavid du Colombier  *
217*219b2ee8SDavid du Colombier  * A non-NULL address refers to a pointer, which is reallocated and
218*219b2ee8SDavid du Colombier  * then reinitialized. A NULL address implies a non-NULL value field
219*219b2ee8SDavid du Colombier  * and describes a character array that we only reinitialize. The
220*219b2ee8SDavid du Colombier  * length field for an array is the size of that array. The length
221*219b2ee8SDavid du Colombier  * field of a pointer is an increment that's added to the length
222*219b2ee8SDavid du Colombier  * required to store the new pathname string - should help when we
223*219b2ee8SDavid du Colombier  * want to change character arrays to pointers in applications like
224*219b2ee8SDavid du Colombier  * troff.
225*219b2ee8SDavid du Colombier  *
226*219b2ee8SDavid du Colombier  */
227*219b2ee8SDavid du Colombier 
228*219b2ee8SDavid du Colombier     if ( (prefix = DWBhome()) == NULL ) {
229*219b2ee8SDavid du Colombier 	fprintf(stderr, "%s: no DWB home directory\n", prog);
230*219b2ee8SDavid du Colombier 	exit(1);
231*219b2ee8SDavid du Colombier     }	/* End if */
232*219b2ee8SDavid du Colombier 
233*219b2ee8SDavid du Colombier     DWBdebug(opaths, 0);
234*219b2ee8SDavid du Colombier     plen = strlen(prefix);
235*219b2ee8SDavid du Colombier 
236*219b2ee8SDavid du Colombier     for ( ; paths->value != NULL || paths->address != NULL; paths++ ) {
237*219b2ee8SDavid du Colombier 	if ( paths->address == NULL ) {
238*219b2ee8SDavid du Colombier 	    length = 0;
239*219b2ee8SDavid du Colombier 	    value = paths->value;
240*219b2ee8SDavid du Colombier 	} else {
241*219b2ee8SDavid du Colombier 	    length = paths->length;
242*219b2ee8SDavid du Colombier 	    value = *paths->address;
243*219b2ee8SDavid du Colombier 	}   /* End else */
244*219b2ee8SDavid du Colombier 
245*219b2ee8SDavid du Colombier 	length += plen + 1 + strlen(value);	/* +1 is for the '/' */
246*219b2ee8SDavid du Colombier 
247*219b2ee8SDavid du Colombier 	if ( (path = malloc(length+1)) == NULL ) {
248*219b2ee8SDavid du Colombier 	    fprintf(stderr, "%s: can't allocate pathname memory\n", prog);
249*219b2ee8SDavid du Colombier 	    exit(1);
250*219b2ee8SDavid du Colombier 	}   /* End if */
251*219b2ee8SDavid du Colombier 
252*219b2ee8SDavid du Colombier 	if ( *value != '\0' ) {
253*219b2ee8SDavid du Colombier 	    char *eop = prefix;
254*219b2ee8SDavid du Colombier 	    while(*eop++)
255*219b2ee8SDavid du Colombier 		;
256*219b2ee8SDavid du Colombier 	    eop -= 2;
257*219b2ee8SDavid du Colombier 	    if (*value != '/' && *eop != '/') {
258*219b2ee8SDavid du Colombier 		sprintf(path, "%s/%s", prefix, value);
259*219b2ee8SDavid du Colombier 	    } else if (*value == '/' && *eop == '/') {
260*219b2ee8SDavid du Colombier 		value++;
261*219b2ee8SDavid du Colombier 		sprintf(path, "%s%s", prefix, value);
262*219b2ee8SDavid du Colombier 	    } else
263*219b2ee8SDavid du Colombier 		sprintf(path, "%s%s", prefix, value);
264*219b2ee8SDavid du Colombier 	} else
265*219b2ee8SDavid du Colombier 		sprintf(path, "%s", prefix);
266*219b2ee8SDavid du Colombier 
267*219b2ee8SDavid du Colombier 	if ( paths->address == NULL ) {
268*219b2ee8SDavid du Colombier 	    if ( strlen(path) >= paths->length ) {
269*219b2ee8SDavid du Colombier 		fprintf(stderr, "%s: no room for %s\n", prog, path);
270*219b2ee8SDavid du Colombier 		exit(1);
271*219b2ee8SDavid du Colombier 	    }	/* End if */
272*219b2ee8SDavid du Colombier 	    strcpy(paths->value, path);
273*219b2ee8SDavid du Colombier 	    free(path);
274*219b2ee8SDavid du Colombier 	} else *paths->address = path;
275*219b2ee8SDavid du Colombier     }	/* End for */
276*219b2ee8SDavid du Colombier 
277*219b2ee8SDavid du Colombier     DWBdebug(opaths, 1);
278*219b2ee8SDavid du Colombier 
279*219b2ee8SDavid du Colombier }   /* End of DWBinit */
280*219b2ee8SDavid du Colombier 
281*219b2ee8SDavid du Colombier /*****************************************************************************/
282*219b2ee8SDavid du Colombier 
DWBprefix(char * prog,char * path,int length)283*219b2ee8SDavid du Colombier void DWBprefix( char *prog, char *path, int length)
284*219b2ee8SDavid du Colombier {
285*219b2ee8SDavid du Colombier 
286*219b2ee8SDavid du Colombier     char	*home;
287*219b2ee8SDavid du Colombier     char	buf[512];
288*219b2ee8SDavid du Colombier     int		len = strlen(DWBPREFIX);
289*219b2ee8SDavid du Colombier 
290*219b2ee8SDavid du Colombier /*
291*219b2ee8SDavid du Colombier  *
292*219b2ee8SDavid du Colombier  * Replace a leading DWBPREFIX string in path by the current DWBhome().
293*219b2ee8SDavid du Colombier  * Used by programs that pretend to handle .so requests. Assumes path
294*219b2ee8SDavid du Colombier  * is an array with room for length characters. The implementation is
295*219b2ee8SDavid du Colombier  * not great, but should be good enough for now. Also probably should
296*219b2ee8SDavid du Colombier  * have DWBhome() only do the lookup once, and remember the value if
297*219b2ee8SDavid du Colombier  * called again.
298*219b2ee8SDavid du Colombier  *
299*219b2ee8SDavid du Colombier  */
300*219b2ee8SDavid du Colombier 
301*219b2ee8SDavid du Colombier     if ( strncmp(path, DWBPREFIX, len) == 0 ) {
302*219b2ee8SDavid du Colombier 	if ( (home = DWBhome()) != NULL ) {
303*219b2ee8SDavid du Colombier 	    if ( strlen(home) + strlen(path+len) < length ) {
304*219b2ee8SDavid du Colombier 		sprintf(buf, "%s%s", home, path+len);
305*219b2ee8SDavid du Colombier 		strcpy(path, buf);		/* assuming there's room in path */
306*219b2ee8SDavid du Colombier 	    } else fprintf(stderr, "%s: no room to grow path %s", prog, path);
307*219b2ee8SDavid du Colombier 	}   /* End if */
308*219b2ee8SDavid du Colombier     }	/* End if */
309*219b2ee8SDavid du Colombier 
310*219b2ee8SDavid du Colombier }   /* End of DWBprefix */
311*219b2ee8SDavid du Colombier 
312*219b2ee8SDavid du Colombier /*****************************************************************************/
313*219b2ee8SDavid du Colombier 
314