186d7f5d3SJohn Marino /* Copyright 1988,1990,1993,1994 by Paul Vixie
286d7f5d3SJohn Marino * All rights reserved
386d7f5d3SJohn Marino *
486d7f5d3SJohn Marino * Distribute freely, except: don't remove my name from the source or
586d7f5d3SJohn Marino * documentation (don't take credit for my work), mark your changes (don't
686d7f5d3SJohn Marino * get me blamed for your possible bugs), don't alter or remove this
786d7f5d3SJohn Marino * notice. May be sold if buildable source is provided to buyer. No
886d7f5d3SJohn Marino * warrantee of any kind, express or implied, is included with this
986d7f5d3SJohn Marino * software; use at your own risk, responsibility for damages (if any) to
1086d7f5d3SJohn Marino * anyone resulting from the use of this software rests entirely with the
1186d7f5d3SJohn Marino * user.
1286d7f5d3SJohn Marino *
1386d7f5d3SJohn Marino * Send bug reports, bug fixes, enhancements, requests, flames, etc., and
1486d7f5d3SJohn Marino * I'll try to keep a version up to date. I can be reached as follows:
1586d7f5d3SJohn Marino * Paul Vixie <paul@vix.com> uunet!decwrl!vixie!paul
1686d7f5d3SJohn Marino *
1786d7f5d3SJohn Marino * $FreeBSD: src/usr.sbin/cron/lib/env.c,v 1.7.2.1 2000/07/01 10:35:07 ps Exp $
1886d7f5d3SJohn Marino * $DragonFly: src/usr.sbin/cron/lib/env.c,v 1.5 2004/12/18 22:48:03 swildner Exp $
1986d7f5d3SJohn Marino */
2086d7f5d3SJohn Marino
2186d7f5d3SJohn Marino #include "cron.h"
2286d7f5d3SJohn Marino
2386d7f5d3SJohn Marino
2486d7f5d3SJohn Marino char **
env_init(void)2586d7f5d3SJohn Marino env_init(void)
2686d7f5d3SJohn Marino {
2786d7f5d3SJohn Marino char **p;
2886d7f5d3SJohn Marino
2986d7f5d3SJohn Marino p = malloc(sizeof(char *));
3086d7f5d3SJohn Marino if (p)
3186d7f5d3SJohn Marino p[0] = NULL;
3286d7f5d3SJohn Marino return (p);
3386d7f5d3SJohn Marino }
3486d7f5d3SJohn Marino
3586d7f5d3SJohn Marino
3686d7f5d3SJohn Marino void
env_free(char ** envp)3786d7f5d3SJohn Marino env_free(char **envp)
3886d7f5d3SJohn Marino {
3986d7f5d3SJohn Marino char **p;
4086d7f5d3SJohn Marino
4186d7f5d3SJohn Marino for (p = envp; *p; p++)
4286d7f5d3SJohn Marino free(*p);
4386d7f5d3SJohn Marino free(envp);
4486d7f5d3SJohn Marino }
4586d7f5d3SJohn Marino
4686d7f5d3SJohn Marino
4786d7f5d3SJohn Marino char **
env_copy(char ** envp)4886d7f5d3SJohn Marino env_copy(char **envp)
4986d7f5d3SJohn Marino {
5086d7f5d3SJohn Marino int count, i;
5186d7f5d3SJohn Marino char **p;
5286d7f5d3SJohn Marino
5386d7f5d3SJohn Marino for (count = 0; envp[count] != NULL; count++)
5486d7f5d3SJohn Marino ;
5586d7f5d3SJohn Marino p = (char **) malloc((count+1) * sizeof(char *)); /* 1 for the NULL */
5686d7f5d3SJohn Marino if (p == NULL) {
5786d7f5d3SJohn Marino errno = ENOMEM;
5886d7f5d3SJohn Marino return NULL;
5986d7f5d3SJohn Marino }
6086d7f5d3SJohn Marino for (i = 0; i < count; i++)
6186d7f5d3SJohn Marino if ((p[i] = strdup(envp[i])) == NULL) {
6286d7f5d3SJohn Marino while (--i >= 0)
6386d7f5d3SJohn Marino free(p[i]);
6486d7f5d3SJohn Marino free(p);
6586d7f5d3SJohn Marino errno = ENOMEM;
6686d7f5d3SJohn Marino return NULL;
6786d7f5d3SJohn Marino }
6886d7f5d3SJohn Marino p[count] = NULL;
6986d7f5d3SJohn Marino return (p);
7086d7f5d3SJohn Marino }
7186d7f5d3SJohn Marino
7286d7f5d3SJohn Marino
7386d7f5d3SJohn Marino char **
env_set(char ** envp,char * envstr)7486d7f5d3SJohn Marino env_set(char **envp, char *envstr)
7586d7f5d3SJohn Marino {
7686d7f5d3SJohn Marino int count, found;
7786d7f5d3SJohn Marino char **p;
7886d7f5d3SJohn Marino char *q;
7986d7f5d3SJohn Marino
8086d7f5d3SJohn Marino /*
8186d7f5d3SJohn Marino * count the number of elements, including the null pointer;
8286d7f5d3SJohn Marino * also set 'found' to -1 or index of entry if already in here.
8386d7f5d3SJohn Marino */
8486d7f5d3SJohn Marino found = -1;
8586d7f5d3SJohn Marino for (count = 0; envp[count] != NULL; count++) {
8686d7f5d3SJohn Marino if (!strcmp_until(envp[count], envstr, '='))
8786d7f5d3SJohn Marino found = count;
8886d7f5d3SJohn Marino }
8986d7f5d3SJohn Marino count++; /* for the NULL */
9086d7f5d3SJohn Marino
9186d7f5d3SJohn Marino if (found != -1) {
9286d7f5d3SJohn Marino /*
9386d7f5d3SJohn Marino * it exists already, so just free the existing setting,
9486d7f5d3SJohn Marino * save our new one there, and return the existing array.
9586d7f5d3SJohn Marino */
9686d7f5d3SJohn Marino q = envp[found];
9786d7f5d3SJohn Marino if ((envp[found] = strdup(envstr)) == NULL) {
9886d7f5d3SJohn Marino envp[found] = q;
9986d7f5d3SJohn Marino /* XXX env_free(envp); */
10086d7f5d3SJohn Marino errno = ENOMEM;
10186d7f5d3SJohn Marino return NULL;
10286d7f5d3SJohn Marino }
10386d7f5d3SJohn Marino free(q);
10486d7f5d3SJohn Marino return (envp);
10586d7f5d3SJohn Marino }
10686d7f5d3SJohn Marino
10786d7f5d3SJohn Marino /*
10886d7f5d3SJohn Marino * it doesn't exist yet, so resize the array, move null pointer over
10986d7f5d3SJohn Marino * one, save our string over the old null pointer, and return resized
11086d7f5d3SJohn Marino * array.
11186d7f5d3SJohn Marino */
11286d7f5d3SJohn Marino p = (char **) realloc((void *) envp,
11386d7f5d3SJohn Marino (unsigned) ((count+1) * sizeof(char *)));
11486d7f5d3SJohn Marino if (p == NULL) {
11586d7f5d3SJohn Marino /* XXX env_free(envp); */
11686d7f5d3SJohn Marino errno = ENOMEM;
11786d7f5d3SJohn Marino return NULL;
11886d7f5d3SJohn Marino }
11986d7f5d3SJohn Marino p[count] = p[count-1];
12086d7f5d3SJohn Marino if ((p[count-1] = strdup(envstr)) == NULL) {
12186d7f5d3SJohn Marino env_free(p);
12286d7f5d3SJohn Marino errno = ENOMEM;
12386d7f5d3SJohn Marino return NULL;
12486d7f5d3SJohn Marino }
12586d7f5d3SJohn Marino return (p);
12686d7f5d3SJohn Marino }
12786d7f5d3SJohn Marino
12886d7f5d3SJohn Marino
12986d7f5d3SJohn Marino /* return ERR = end of file
13086d7f5d3SJohn Marino * FALSE = not an env setting (file was repositioned)
13186d7f5d3SJohn Marino * TRUE = was an env setting
13286d7f5d3SJohn Marino */
13386d7f5d3SJohn Marino int
load_env(char * envstr,FILE * f)13486d7f5d3SJohn Marino load_env(char *envstr, FILE *f)
13586d7f5d3SJohn Marino {
13686d7f5d3SJohn Marino long filepos;
13786d7f5d3SJohn Marino int fileline;
13886d7f5d3SJohn Marino char name[MAX_ENVSTR], val[MAX_ENVSTR];
13986d7f5d3SJohn Marino int fields;
14086d7f5d3SJohn Marino
14186d7f5d3SJohn Marino filepos = ftell(f);
14286d7f5d3SJohn Marino fileline = LineNumber;
14386d7f5d3SJohn Marino skip_comments(f);
14486d7f5d3SJohn Marino if (EOF == get_string(envstr, MAX_ENVSTR, f, "\n"))
14586d7f5d3SJohn Marino return (ERR);
14686d7f5d3SJohn Marino
14786d7f5d3SJohn Marino Debug(DPARS, ("load_env, read <%s>\n", envstr))
14886d7f5d3SJohn Marino
14986d7f5d3SJohn Marino name[0] = val[0] = '\0';
15086d7f5d3SJohn Marino fields = sscanf(envstr, "%[^ =] = %[^\n#]", name, val);
15186d7f5d3SJohn Marino if (fields != 2) {
15286d7f5d3SJohn Marino Debug(DPARS, ("load_env, not 2 fields (%d)\n", fields))
15386d7f5d3SJohn Marino fseek(f, filepos, 0);
15486d7f5d3SJohn Marino Set_LineNum(fileline);
15586d7f5d3SJohn Marino return (FALSE);
15686d7f5d3SJohn Marino }
15786d7f5d3SJohn Marino
15886d7f5d3SJohn Marino /* 2 fields from scanf; looks like an env setting
15986d7f5d3SJohn Marino */
16086d7f5d3SJohn Marino
16186d7f5d3SJohn Marino /*
16286d7f5d3SJohn Marino * process value string
16386d7f5d3SJohn Marino */
16486d7f5d3SJohn Marino /*local*/{
16586d7f5d3SJohn Marino int len = strdtb(val);
16686d7f5d3SJohn Marino
16786d7f5d3SJohn Marino if (len >= 2) {
16886d7f5d3SJohn Marino if (val[0] == '\'' || val[0] == '"') {
16986d7f5d3SJohn Marino if (val[len-1] == val[0]) {
17086d7f5d3SJohn Marino val[len-1] = '\0';
17186d7f5d3SJohn Marino strcpy(val, val+1);
17286d7f5d3SJohn Marino }
17386d7f5d3SJohn Marino }
17486d7f5d3SJohn Marino }
17586d7f5d3SJohn Marino }
17686d7f5d3SJohn Marino
17786d7f5d3SJohn Marino if (strlen(name) + 1 + strlen(val) >= MAX_ENVSTR-1)
17886d7f5d3SJohn Marino return (FALSE);
17986d7f5d3SJohn Marino sprintf(envstr, "%s=%s", name, val);
18086d7f5d3SJohn Marino Debug(DPARS, ("load_env, <%s> <%s> -> <%s>\n", name, val, envstr))
18186d7f5d3SJohn Marino return (TRUE);
18286d7f5d3SJohn Marino }
18386d7f5d3SJohn Marino
18486d7f5d3SJohn Marino
18586d7f5d3SJohn Marino char *
env_get(char * name,char ** envp)18686d7f5d3SJohn Marino env_get(char *name, char **envp)
18786d7f5d3SJohn Marino {
18886d7f5d3SJohn Marino int len;
18986d7f5d3SJohn Marino char *p, *q;
19086d7f5d3SJohn Marino
19186d7f5d3SJohn Marino len = strlen(name);
19286d7f5d3SJohn Marino while ((p = *envp++)) {
19386d7f5d3SJohn Marino if (!(q = strchr(p, '=')))
19486d7f5d3SJohn Marino continue;
19586d7f5d3SJohn Marino if ((q - p) == len && !strncmp(p, name, len))
19686d7f5d3SJohn Marino return (q+1);
19786d7f5d3SJohn Marino }
19886d7f5d3SJohn Marino return (NULL);
19986d7f5d3SJohn Marino }
200