2 * $OpenBSD: util.c,v 1.32 2006/03/11 19:41:30 otto Exp $
3 * $DragonFly: src/usr.bin/patch/util.c,v 1.9 2007/09/29 23:11:10 swildner Exp $
4 * $NetBSD: util.c,v 1.24 2008/09/19 18:33:34 joerg Exp $
8 * patch - a program to apply diffs to original files
10 * Copyright 1986, Larry Wall
12 * Redistribution and use in source and binary forms, with or without
13 * modification, are permitted provided that the following condition is met:
14 * 1. Redistributions of source code must retain the above copyright notice,
15 * this condition and the following disclaimer.
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND ANY
18 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
19 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
20 * DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE FOR
21 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
23 * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
24 * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
29 * -C option added in 1998, original code by Marc Espie, based on FreeBSD
33 #include <sys/cdefs.h>
35 #include <sys/param.h>
52 #include "backupfile.h"
53 #include "pathnames.h"
55 /* Rename a file, copying it if necessary. */
58 move_file(const char *from
, const char *to
)
68 say("Moving %s to stdout.\n", from
);
70 fromfd
= open(from
, O_RDONLY
);
72 pfatal("internal error, can't reopen %s", from
);
73 while ((i
= read(fromfd
, buf
, buf_len
)) > 0)
74 if (write(STDOUT_FILENO
, buf
, i
) != i
)
75 pfatal("write failed");
79 if (backup_file(to
) < 0) {
80 say("Can't backup %s, output is in %s: %s\n", to
, from
,
86 say("Moving %s to %s.\n", from
, to
);
88 if (rename(from
, to
) < 0) {
89 if (errno
!= EXDEV
|| copy_file(from
, to
) < 0) {
90 say("Can't create %s, output is in %s: %s\n",
91 to
, from
, strerror(errno
));
98 /* Backup the original file. */
101 backup_file(const char *orig
)
103 struct stat filestat
;
104 char bakname
[MAXPATHLEN
], *s
, *simplename
;
108 if (backup_type
== none
|| stat(orig
, &filestat
) != 0)
109 return 0; /* nothing to do */
111 * If the user used zero prefixes or suffixes, then
112 * he doesn't want backups. Yet we have to remove
113 * orig to break possible hardlinks.
115 if ((origprae
&& *origprae
== 0) || *simple_backup_suffix
== 0) {
119 orig_device
= filestat
.st_dev
;
120 orig_inode
= filestat
.st_ino
;
123 if (strlcpy(bakname
, origprae
, sizeof(bakname
)) >= sizeof(bakname
) ||
124 strlcat(bakname
, orig
, sizeof(bakname
)) >= sizeof(bakname
))
125 fatal("filename %s too long for buffer\n", origprae
);
127 if ((s
= find_backup_file_name(orig
)) == NULL
)
128 fatal("out of memory\n");
129 if (strlcpy(bakname
, s
, sizeof(bakname
)) >= sizeof(bakname
))
130 fatal("filename %s too long for buffer\n", s
);
134 if ((simplename
= strrchr(bakname
, '/')) != NULL
)
135 simplename
= simplename
+ 1;
137 simplename
= bakname
;
140 * Find a backup name that is not the same file. Change the
141 * first lowercase char into uppercase; if that isn't
142 * sufficient, chop off the first char and try again.
144 while (stat(bakname
, &filestat
) == 0 &&
145 orig_device
== filestat
.st_dev
&& orig_inode
== filestat
.st_ino
) {
146 /* Skip initial non-lowercase chars. */
147 for (s
= simplename
; *s
&& !islower((unsigned char)*s
); s
++)
150 *s
= toupper((unsigned char)*s
);
152 memmove(simplename
, simplename
+ 1,
153 strlen(simplename
+ 1) + 1);
157 say("Moving %s to %s.\n", orig
, bakname
);
159 if (rename(orig
, bakname
) < 0) {
160 if (errno
!= EXDEV
|| copy_file(orig
, bakname
) < 0)
170 copy_file(const char *from
, const char *to
)
175 tofd
= open(to
, O_CREAT
|O_TRUNC
|O_WRONLY
, 0666);
178 fromfd
= open(from
, O_RDONLY
, 0);
180 pfatal("internal error, can't reopen %s", from
);
181 while ((i
= read(fromfd
, buf
, buf_len
)) > 0)
182 if (write(tofd
, buf
, i
) != i
)
183 pfatal("write to %s failed", to
);
190 * Allocate a unique area for a string.
193 savestr(const char *s
)
204 fatal("out of memory\n");
210 * Vanilla terminal output (buffered).
213 say(const char *fmt
, ...)
218 vfprintf(stderr
, fmt
, ap
);
224 * Terminal output, pun intended.
227 fatal(const char *fmt
, ...)
232 fprintf(stderr
, "patch: **** ");
233 vfprintf(stderr
, fmt
, ap
);
239 * Say something from patch, something from the system, then silence . . .
242 pfatal(const char *fmt
, ...)
247 fprintf(stderr
, "patch: **** ");
249 vfprintf(stderr
, fmt
, ap
);
251 fprintf(stderr
, ": %s\n", strerror(errnum
));
256 * Get a response from the user via /dev/tty
259 ask(const char *fmt
, ...)
263 static int ttyfd
= -1;
266 vfprintf(stdout
, fmt
, ap
);
270 ttyfd
= open(_PATH_TTY
, O_RDONLY
);
272 if ((nr
= read(ttyfd
, buf
, buf_len
)) > 0 &&
276 if (ttyfd
< 0 || nr
<= 0) {
277 /* no tty or error reading, pretend user entered 'return' */
284 * How to handle certain events when not in a critical region.
287 set_signals(int reset
)
289 static sig_t hupval
, intval
;
292 hupval
= signal(SIGHUP
, SIG_IGN
);
293 if (hupval
!= SIG_IGN
)
295 intval
= signal(SIGINT
, SIG_IGN
);
296 if (intval
!= SIG_IGN
)
299 signal(SIGHUP
, hupval
);
300 signal(SIGINT
, intval
);
304 * How to handle certain events when in a critical region.
309 signal(SIGHUP
, SIG_IGN
);
310 signal(SIGINT
, SIG_IGN
);
314 * Make sure we'll have the directories to create a file. If `striplast' is
315 * true, ignore the last element of `filename'.
319 makedirs(const char *filename
, bool striplast
)
323 if ((tmpbuf
= strdup(filename
)) == NULL
)
324 fatal("out of memory\n");
327 char *s
= strrchr(tmpbuf
, '/');
329 return; /* nothing to be done */
332 if (mkpath(tmpbuf
) != 0)
333 pfatal("creation of %s failed", tmpbuf
);
338 * Make filenames more reasonable.
341 fetchname(const char *at
, bool *exists
, int strip_leading
)
343 char *fullname
, *name
, *t
;
345 struct stat filestat
;
347 if (at
== NULL
|| *at
== '\0')
349 while (isspace((unsigned char)*at
))
353 say("fetchname %s %d\n", at
, strip_leading
);
355 /* So files can be created by diffing against /dev/null. */
356 if (strnEQ(at
, _PATH_DEVNULL
, sizeof(_PATH_DEVNULL
) - 1))
358 name
= fullname
= t
= savestr(at
);
360 tab
= strchr(t
, '\t') != NULL
;
361 /* Strip off up to `strip_leading' path components and NUL terminate. */
362 for (sleading
= strip_leading
; *t
!= '\0' && ((tab
&& *t
!= '\t') ||
363 !isspace((unsigned char)*t
)); t
++) {
364 if (t
[0] == '/' && t
[1] != '/' && t
[1] != '\0')
371 * If no -p option was given (957 is the default value!), we were
372 * given a relative pathname, and the leading directories that we
373 * just stripped off all exist, put them back on.
375 if (strip_leading
== 957 && name
!= fullname
&& *fullname
!= '/') {
377 if (stat(fullname
, &filestat
) == 0 && S_ISDIR(filestat
.st_mode
)) {
382 name
= savestr(name
);
385 *exists
= stat(name
, &filestat
) == 0;
390 * Takes the name returned by fetchname and looks in RCS/SCCS directories
391 * for a checked in version.
394 checked_in(char *file
)
396 char *filebase
, *filedir
, tmpbuf
[MAXPATHLEN
];
397 struct stat filestat
;
399 filebase
= basename(file
);
400 filedir
= dirname(file
);
402 #define try(f, a1, a2, a3) \
403 (snprintf(tmpbuf, sizeof tmpbuf, f, a1, a2, a3), stat(tmpbuf, &filestat) == 0)
405 if (try("%s/RCS/%s%s", filedir
, filebase
, RCSSUFFIX
) ||
406 try("%s/RCS/%s%s", filedir
, filebase
, "") ||
407 try("%s/%s%s", filedir
, filebase
, RCSSUFFIX
) ||
408 try("%s/SCCS/%s%s", filedir
, SCCSPREFIX
, filebase
) ||
409 try("%s/%s%s", filedir
, SCCSPREFIX
, filebase
))
418 fprintf(stderr
, "Patch version 2.0-12u8-NetBSD\n");
419 my_exit(EXIT_SUCCESS
);