| /* |
| * |
| * 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 */ |
| if ( home != NULL ) |
| 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 */ |
| |
| /*****************************************************************************/ |
| |