/* * * Pathname management routines for DWB C programs. * * Applications should initialize a dwbinit array with the string * pointers and arrays that need to be updated, and then hand that * array to DWBinit before much else happens in their main program. * DWBinit calls DWBhome to get the current home directory. DWBhome * uses the last definition of DWBENV (usually "DWBHOME") in file * DWBCONFIG (e.g., /usr/lib/dwb3.4) or the value assigned to that * variable in the environment if the DWBCONFIG file doesn't exist, * can't be read, or doesn't define DWBENV. * * DWBCONFIG must be a simple shell script - comments, a definition * of DWBHOME, and perhaps an export or echo is about all that's * allowed. The parsing in DWBhome is simple and makes no attempt * to duplicate the shell. It only looks for DWBHOME= as the first * non-white space string on a line, so * * # * # A sample DWBCONFIG shell script * # * * DWBHOME=/usr/add-on/dwb3.4 * export DWBHOME * * means DWBhome would return "/usr/add-on/dwb3.4" for the DWB home * directory. A DWBCONFIG file means there can only be one working * copy of a DWB release on a system, which seems like a good idea. * Using DWBCONFIG also means programs will always include correct * versions of files (e.g., prologues or macro packages). * * Relying on an environment variable guarantees nothing. You could * execute a version of dpost, but your environment might point at * incorrect font tables or prologues. Despite the obvious problems * we've also implemented an environment variable approach, but it's * only used if there's no DWBCONFIG file. * * DWBinit calls DWBhome to get the DWB home directory prefix and * then marches through its dwbinit argument, removing the default * home directory and prepending the new home. DWBinit stops when * it reaches an element that has NULL for its address and value * fields. Pointers in a dwbinit array are reallocated and properly * initialized; arrays are simply reinitialized if there's room. * All pathnames that are to be adjusted should be relative. For * example, * * char *fontdir = "lib/font"; * char xyzzy[25] = "etc/xyzzy"; * * would be represented in a dwbinit array as, * * dwbinit allpaths[] = { * &fontdir, NULL, 0, * NULL, xyzzy, sizeof(xyzzy), * NULL, NULL, 0 * }; * * The last element must have NULL entries for the address and * value fields. The main() routine would then do, * * #include "dwbinit.h" * * main() { * * DWBinit("program name", allpaths); * ... * } * * Debugging is enabled if DWBDEBUG is in the environment and has * the value ON. Output is occasionally useful and probably should * be documented. * */ #include <u.h> #include <stdio.h> #include <ctype.h> #include <string.h> #include <stdlib.h> #include "dwbinit.h" #ifndef DWBCONFIG #define DWBCONFIG "/dev/null" #endif #ifndef DWBENV #define DWBENV "DWBHOME" #endif #ifndef DWBHOME #define DWBHOME "" #endif #ifndef DWBDEBUG #define DWBDEBUG "DWBDEBUG" #endif #ifndef DWBPREFIX #define DWBPREFIX "\\*(.P" #endif /*****************************************************************************/ void DWBdebug(dwbinit *ptr, int level) { char *path; char *home; static char *debug = NULL; /* * * Debugging output, but only if DWBDEBUG is defined to be ON in the * environment. Dumps general info the first time through. * */ if ( debug == NULL && (debug = getenv(DWBDEBUG)) == NULL ) debug = "OFF"; if ( strcmp(debug, "ON") == 0 ) { if ( level == 0 ) { fprintf(stderr, "Environment variable: %s\n", DWBENV); fprintf(stderr, "Configuration file: %s\n", DWBCONFIG); fprintf(stderr, "Default home: %s\n", DWBHOME); if ( (home = DWBhome()) != NULL ) fprintf(stderr, "Current home: %s\n", home); } /* End if */ fprintf(stderr, "\n%s pathnames:\n", level == 0 ? "Original" : "Final"); for ( ; ptr->value != NULL || ptr->address != NULL; ptr++ ) { if ( (path = ptr->value) == NULL ) { path = *ptr->address; fprintf(stderr, " pointer: %s\n", path); } else fprintf(stderr, " array[%d]: %s\n", ptr->length, path); if ( level == 0 && *path == '/' ) fprintf(stderr, " WARNING - absolute path\n"); } /* End for */ } /* End if */ } /* End of DWBdebug */ /*****************************************************************************/ extern char *unsharp(char*); char *DWBhome(void) { FILE *fp; char *ptr; char *path; int len; char buf[200]; char *home = NULL; /* * * Return the DWB home directory. Uses the last definition of DWBENV * (usually "DWBHOME") in file DWBCONFIG (perhaps /usr/lib/dwb3.4) or * the value assigned to the variable named by the DWBENV string in * the environment if DWBCONFIG doesn't exist or doesn't define DWBENV. * Skips the file lookup if DWBCONFIG can't be read. Returns NULL if * there's no home directory. * */ if ( (fp = fopen(DWBCONFIG, "r")) != NULL ) { len = strlen(DWBENV); while ( fgets(buf, sizeof(buf), fp) != NULL ) { for ( ptr = buf; isspace((uchar)*ptr); ptr++ ) ; if ( strncmp(ptr, DWBENV, len) == 0 && *(ptr+len) == '=' ) { path = ptr + len + 1; for ( ptr = path; !isspace((uchar)*ptr) && *ptr != ';'; ptr++ ) ; *ptr = '\0'; if ( home != NULL ) free(home); if ( (home = malloc(strlen(path)+1)) != NULL ) strcpy(home, path); } /* End if */ } /* End while */ fclose(fp); } /* End if */ if ( home == NULL ) { if ( (home = getenv(DWBENV)) == NULL ) { if ( (home = DWBHOME) == NULL || *home == '\0' || *home == ' ' ) home = NULL; } /* End if */ home = unsharp(home); } /* End if */ while (home && *home == '/' && *(home +1) == '/') /* remove extra slashes */ home++; return(home); } /* End of DWBhome */ /*****************************************************************************/ void DWBinit(char *prog, dwbinit *paths) { char *prefix; char *value; char *path; int plen; int length; dwbinit *opaths = paths; /* * * Adjust the pathnames listed in paths, using the home directory * returned by DWBhome(). Stops when it reaches an element that has * NULL address and value fields. Assumes pathnames are relative, * but changes everything. DWBdebug issues a warning if an original * path begins with a /. * * A non-NULL address refers to a pointer, which is reallocated and * then reinitialized. A NULL address implies a non-NULL value field * and describes a character array that we only reinitialize. The * length field for an array is the size of that array. The length * field of a pointer is an increment that's added to the length * required to store the new pathname string - should help when we * want to change character arrays to pointers in applications like * troff. * */ if ( (prefix = DWBhome()) == NULL ) { fprintf(stderr, "%s: no DWB home directory\n", prog); exit(1); } /* End if */ DWBdebug(opaths, 0); plen = strlen(prefix); for ( ; paths->value != NULL || paths->address != NULL; paths++ ) { if ( paths->address == NULL ) { length = 0; value = paths->value; } else { length = paths->length; value = *paths->address; } /* End else */ length += plen + 1 + strlen(value); /* +1 is for the '/' */ if ( (path = malloc(length+1)) == NULL ) { fprintf(stderr, "%s: can't allocate pathname memory\n", prog); exit(1); } /* End if */ if ( *value != '\0' ) { char *eop = prefix; while(*eop++) ; eop -= 2; if (*value != '/' && *eop != '/') { sprintf(path, "%s/%s", prefix, value); } else if (*value == '/' && *eop == '/') { value++; sprintf(path, "%s%s", prefix, value); } else sprintf(path, "%s%s", prefix, value); } else sprintf(path, "%s", prefix); if ( paths->address == NULL ) { if ( strlen(path) >= paths->length ) { fprintf(stderr, "%s: no room for %s\n", prog, path); exit(1); } /* End if */ strcpy(paths->value, path); free(path); } else *paths->address = path; } /* End for */ DWBdebug(opaths, 1); } /* End of DWBinit */ /*****************************************************************************/ void DWBprefix( char *prog, char *path, int length) { char *home; char buf[512]; int len = strlen(DWBPREFIX); /* * * Replace a leading DWBPREFIX string in path by the current DWBhome(). * Used by programs that pretend to handle .so requests. Assumes path * is an array with room for length characters. The implementation is * not great, but should be good enough for now. Also probably should * have DWBhome() only do the lookup once, and remember the value if * called again. * */ if ( strncmp(path, DWBPREFIX, len) == 0 ) { if ( (home = DWBhome()) != NULL ) { if ( strlen(home) + strlen(path+len) < length ) { sprintf(buf, "%s%s", home, path+len); strcpy(path, buf); /* assuming there's room in path */ } else fprintf(stderr, "%s: no room to grow path %s", prog, path); } /* End if */ } /* End if */ } /* End of DWBprefix */ /*****************************************************************************/