]> andersk Git - openssh.git/blob - openbsd-compat/mktemp.c
- (djm) [auth-pam.c clientloop.c includes.h monitor.c session.c]
[openssh.git] / openbsd-compat / mktemp.c
1 /* THIS FILE HAS BEEN MODIFIED FROM THE ORIGINAL OPENBSD SOURCE */
2 /* Changes: Removed mktemp */
3
4 /*      $OpenBSD: mktemp.c,v 1.19 2005/08/08 08:05:36 espie Exp $ */
5 /*
6  * Copyright (c) 1987, 1993
7  *      The Regents of the University of California.  All rights reserved.
8  *
9  * Redistribution and use in source and binary forms, with or without
10  * modification, are permitted provided that the following conditions
11  * are met:
12  * 1. Redistributions of source code must retain the above copyright
13  *    notice, this list of conditions and the following disclaimer.
14  * 2. Redistributions in binary form must reproduce the above copyright
15  *    notice, this list of conditions and the following disclaimer in the
16  *    documentation and/or other materials provided with the distribution.
17  * 3. Neither the name of the University nor the names of its contributors
18  *    may be used to endorse or promote products derived from this software
19  *    without specific prior written permission.
20  *
21  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
22  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
25  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
26  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
27  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
28  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
29  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
30  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
31  * SUCH DAMAGE.
32  */
33
34 /* OPENBSD ORIGINAL: lib/libc/stdio/mktemp.c */
35
36 #include "includes.h"
37
38 #include <sys/types.h>
39 #include <sys/stat.h>
40 #include <ctype.h>
41
42 #if !defined(HAVE_MKDTEMP) || defined(HAVE_STRICT_MKSTEMP)
43
44 static int _gettemp(char *, int *, int, int);
45
46 int
47 mkstemps(char *path, int slen)
48 {
49         int fd;
50
51         return (_gettemp(path, &fd, 0, slen) ? fd : -1);
52 }
53
54 int
55 mkstemp(char *path)
56 {
57         int fd;
58
59         return (_gettemp(path, &fd, 0, 0) ? fd : -1);
60 }
61
62 char *
63 mkdtemp(char *path)
64 {
65         return(_gettemp(path, (int *)NULL, 1, 0) ? path : (char *)NULL);
66 }
67
68 static int
69 _gettemp(path, doopen, domkdir, slen)
70         char *path;
71         register int *doopen;
72         int domkdir;
73         int slen;
74 {
75         register char *start, *trv, *suffp;
76         struct stat sbuf;
77         int rval;
78         pid_t pid;
79
80         if (doopen && domkdir) {
81                 errno = EINVAL;
82                 return(0);
83         }
84
85         for (trv = path; *trv; ++trv)
86                 ;
87         trv -= slen;
88         suffp = trv;
89         --trv;
90         if (trv < path) {
91                 errno = EINVAL;
92                 return (0);
93         }
94         pid = getpid();
95         while (trv >= path && *trv == 'X' && pid != 0) {
96                 *trv-- = (pid % 10) + '0';
97                 pid /= 10;
98         }
99         while (trv >= path && *trv == 'X') {
100                 char c;
101
102                 pid = (arc4random() & 0xffff) % (26+26);
103                 if (pid < 26)
104                         c = pid + 'A';
105                 else
106                         c = (pid - 26) + 'a';
107                 *trv-- = c;
108         }
109         start = trv + 1;
110
111         /*
112          * check the target directory; if you have six X's and it
113          * doesn't exist this runs for a *very* long time.
114          */
115         if (doopen || domkdir) {
116                 for (;; --trv) {
117                         if (trv <= path)
118                                 break;
119                         if (*trv == '/') {
120                                 *trv = '\0';
121                                 rval = stat(path, &sbuf);
122                                 *trv = '/';
123                                 if (rval != 0)
124                                         return(0);
125                                 if (!S_ISDIR(sbuf.st_mode)) {
126                                         errno = ENOTDIR;
127                                         return(0);
128                                 }
129                                 break;
130                         }
131                 }
132         }
133
134         for (;;) {
135                 if (doopen) {
136                         if ((*doopen =
137                             open(path, O_CREAT|O_EXCL|O_RDWR, 0600)) >= 0)
138                                 return(1);
139                         if (errno != EEXIST)
140                                 return(0);
141                 } else if (domkdir) {
142                         if (mkdir(path, 0700) == 0)
143                                 return(1);
144                         if (errno != EEXIST)
145                                 return(0);
146                 } else if (lstat(path, &sbuf))
147                         return(errno == ENOENT ? 1 : 0);
148
149                 /* tricky little algorithm for backward compatibility */
150                 for (trv = start;;) {
151                         if (!*trv)
152                                 return (0);
153                         if (*trv == 'Z') {
154                                 if (trv == suffp)
155                                         return (0);
156                                 *trv++ = 'a';
157                         } else {
158                                 if (isdigit(*trv))
159                                         *trv = 'a';
160                                 else if (*trv == 'z')   /* inc from z to A */
161                                         *trv = 'A';
162                                 else {
163                                         if (trv == suffp)
164                                                 return (0);
165                                         ++*trv;
166                                 }
167                                 break;
168                         }
169                 }
170         }
171         /*NOTREACHED*/
172 }
173
174 #endif /* !defined(HAVE_MKDTEMP) || defined(HAVE_STRICT_MKSTEMP) */
This page took 0.047638 seconds and 5 git commands to generate.