xref: /netbsd-src/usr.bin/vndcompress/utils.c (revision cab898d0ae5e9b05a490b2f7733c27595d447806)
1*cab898d0Sriastradh /*	$NetBSD: utils.c,v 1.6 2017/03/21 13:56:38 riastradh Exp $	*/
2d99bda47Sriastradh 
3d99bda47Sriastradh /*-
4d99bda47Sriastradh  * Copyright (c) 2013 The NetBSD Foundation, Inc.
5d99bda47Sriastradh  * All rights reserved.
6d99bda47Sriastradh  *
7d99bda47Sriastradh  * This code is derived from software contributed to The NetBSD Foundation
8d99bda47Sriastradh  * by Taylor R. Campbell.
9d99bda47Sriastradh  *
10d99bda47Sriastradh  * Redistribution and use in source and binary forms, with or without
11d99bda47Sriastradh  * modification, are permitted provided that the following conditions
12d99bda47Sriastradh  * are met:
13d99bda47Sriastradh  * 1. Redistributions of source code must retain the above copyright
14d99bda47Sriastradh  *    notice, this list of conditions and the following disclaimer.
15d99bda47Sriastradh  * 2. Redistributions in binary form must reproduce the above copyright
16d99bda47Sriastradh  *    notice, this list of conditions and the following disclaimer in the
17d99bda47Sriastradh  *    documentation and/or other materials provided with the distribution.
18d99bda47Sriastradh  *
19d99bda47Sriastradh  * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS
20d99bda47Sriastradh  * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
21d99bda47Sriastradh  * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
22d99bda47Sriastradh  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS
23d99bda47Sriastradh  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
24d99bda47Sriastradh  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
25d99bda47Sriastradh  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
26d99bda47Sriastradh  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
27d99bda47Sriastradh  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
28d99bda47Sriastradh  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
29d99bda47Sriastradh  * POSSIBILITY OF SUCH DAMAGE.
30d99bda47Sriastradh  */
31d99bda47Sriastradh 
32d99bda47Sriastradh #include <sys/cdefs.h>
33*cab898d0Sriastradh __RCSID("$NetBSD: utils.c,v 1.6 2017/03/21 13:56:38 riastradh Exp $");
34d99bda47Sriastradh 
35d99bda47Sriastradh #include <sys/types.h>
36d99bda47Sriastradh 
37d99bda47Sriastradh #include <assert.h>
38d99bda47Sriastradh #include <err.h>
39d99bda47Sriastradh #include <errno.h>
40d99bda47Sriastradh #include <inttypes.h>
41d99bda47Sriastradh #include <limits.h>
426e96c4eaSriastradh #include <signal.h>
43d99bda47Sriastradh #include <stdio.h>
44d99bda47Sriastradh #include <stdlib.h>
45d99bda47Sriastradh #include <string.h>
46d99bda47Sriastradh #include <unistd.h>
47d99bda47Sriastradh 
483e40e9d7Sriastradh #include "common.h"
493e40e9d7Sriastradh 
50d99bda47Sriastradh /* XXX Seems to be missing from <stdio.h>...  */
51d99bda47Sriastradh int	snprintf_ss(char *restrict, size_t, const char *restrict, ...)
52d99bda47Sriastradh 	    __printflike(3, 4);
53d99bda47Sriastradh int	vsnprintf_ss(char *restrict, size_t, const char *restrict, va_list)
54d99bda47Sriastradh 	    __printflike(3, 0);
55d99bda47Sriastradh 
56d99bda47Sriastradh #include "utils.h"
57d99bda47Sriastradh 
58d99bda47Sriastradh /*
59d99bda47Sriastradh  * Read, returning partial data only at end of file.
60d99bda47Sriastradh  */
61d99bda47Sriastradh ssize_t
read_block(int fd,void * buf,size_t len)62*cab898d0Sriastradh read_block(int fd, void *buf, size_t len)
63d99bda47Sriastradh {
64*cab898d0Sriastradh 	char *p = buf;
65*cab898d0Sriastradh 	size_t n = len;
66*cab898d0Sriastradh 	const char *const end __diagused = p + n;
67*cab898d0Sriastradh 	ssize_t nread = 0;
68d99bda47Sriastradh 
69*cab898d0Sriastradh 	while (0 < n && (nread = read(fd, p, n)) != 0) {
70*cab898d0Sriastradh 		if (nread == -1)
71d99bda47Sriastradh 			return -1;
72*cab898d0Sriastradh 		p += MIN(n, (size_t)nread);
73*cab898d0Sriastradh 		n -= MIN(n, (size_t)nread);
74*cab898d0Sriastradh 		assert(p + n == end);
75d99bda47Sriastradh 	}
76d99bda47Sriastradh 
77*cab898d0Sriastradh 	assert(n == 0 || nread == 0); /* complete read or EOF */
78*cab898d0Sriastradh 	return len - n;
79d99bda47Sriastradh }
80d99bda47Sriastradh 
81d99bda47Sriastradh /*
823e40e9d7Sriastradh  * Read from a specified position, returning partial data only at end
833e40e9d7Sriastradh  * of file.
843e40e9d7Sriastradh  */
853e40e9d7Sriastradh ssize_t
pread_block(int fd,void * buf,size_t len,off_t fdpos)86*cab898d0Sriastradh pread_block(int fd, void *buf, size_t len, off_t fdpos)
873e40e9d7Sriastradh {
88*cab898d0Sriastradh 	char *p = buf;
89*cab898d0Sriastradh 	size_t n = len;
90*cab898d0Sriastradh 	const char *const end __diagused = p + n;
91*cab898d0Sriastradh 	ssize_t nread = 0;
923e40e9d7Sriastradh 
935b1eaed1Sriastradh 	assert(0 <= fdpos);
94*cab898d0Sriastradh 	assert(n <= OFF_MAX - (uintmax_t)fdpos);
95*cab898d0Sriastradh 	const off_t endpos __diagused = fdpos + n;
963e40e9d7Sriastradh 
97*cab898d0Sriastradh 	while (0 < n && (nread = pread(fd, p, n, fdpos)) != 0) {
98*cab898d0Sriastradh 		if (nread == -1)
993e40e9d7Sriastradh 			return -1;
100*cab898d0Sriastradh 		fdpos += MIN(n, (size_t)nread);
101*cab898d0Sriastradh 		p += MIN(n, (size_t)nread);
102*cab898d0Sriastradh 		n -= MIN(n, (size_t)nread);
103*cab898d0Sriastradh 		assert(p + n == end);
104*cab898d0Sriastradh 		assert(fdpos + (off_t)n == endpos);
1053e40e9d7Sriastradh 	}
1063e40e9d7Sriastradh 
107*cab898d0Sriastradh 	assert(n == 0 || nread == 0); /* complete read or EOF */
108*cab898d0Sriastradh 	return len - n;
1093e40e9d7Sriastradh }
1103e40e9d7Sriastradh 
1113e40e9d7Sriastradh /*
112d99bda47Sriastradh  * Signal-safe err/warn utilities.  The errno varieties are limited to
113d99bda47Sriastradh  * having no format arguments for reasons of laziness.
114d99bda47Sriastradh  */
115d99bda47Sriastradh 
116d99bda47Sriastradh void
err_ss(int exit_value,const char * msg)117d99bda47Sriastradh err_ss(int exit_value, const char *msg)
118d99bda47Sriastradh {
119d99bda47Sriastradh 	warn_ss(msg);
120d99bda47Sriastradh 	_Exit(exit_value);
121d99bda47Sriastradh }
122d99bda47Sriastradh 
123d99bda47Sriastradh void
errx_ss(int exit_value,const char * format,...)124d99bda47Sriastradh errx_ss(int exit_value, const char *format, ...)
125d99bda47Sriastradh {
126d99bda47Sriastradh 	va_list va;
127d99bda47Sriastradh 
128d99bda47Sriastradh 	va_start(va, format);
129d99bda47Sriastradh 	vwarnx_ss(format, va);
130d99bda47Sriastradh 	va_end(va);
131d99bda47Sriastradh 	_Exit(exit_value);
132d99bda47Sriastradh }
133d99bda47Sriastradh 
134d99bda47Sriastradh void
warn_ss(const char * msg)135d99bda47Sriastradh warn_ss(const char *msg)
136d99bda47Sriastradh {
137d99bda47Sriastradh 	int error = errno;
138d99bda47Sriastradh 
139d99bda47Sriastradh 	warnx_ss("%s: %s", msg, strerror(error));
140d99bda47Sriastradh 
141d99bda47Sriastradh 	errno = error;
142d99bda47Sriastradh }
143d99bda47Sriastradh 
144d99bda47Sriastradh void
warnx_ss(const char * format,...)145d99bda47Sriastradh warnx_ss(const char *format, ...)
146d99bda47Sriastradh {
147d99bda47Sriastradh 	va_list va;
148d99bda47Sriastradh 
149d99bda47Sriastradh 	va_start(va, format);
150d99bda47Sriastradh 	vwarnx_ss(format, va);
151d99bda47Sriastradh 	va_end(va);
152d99bda47Sriastradh }
153d99bda47Sriastradh 
154d99bda47Sriastradh void
vwarnx_ss(const char * format,va_list va)155d99bda47Sriastradh vwarnx_ss(const char *format, va_list va)
156d99bda47Sriastradh {
157d99bda47Sriastradh 	char buf[128];
158d99bda47Sriastradh 
159d99bda47Sriastradh 	(void)strlcpy(buf, getprogname(), sizeof(buf));
160d99bda47Sriastradh 	(void)strlcat(buf, ": ", sizeof(buf));
161d99bda47Sriastradh 
162d99bda47Sriastradh 	const int n = vsnprintf_ss(&buf[strlen(buf)], (sizeof(buf) -
163d99bda47Sriastradh 		strlen(buf)), format, va);
164d99bda47Sriastradh 	if (n <= 0) {
165d99bda47Sriastradh 		const char fallback[] =
166d99bda47Sriastradh 		    "vndcompress: Help!  I'm trapped in a signal handler!\n";
167d99bda47Sriastradh 		(void)write(STDERR_FILENO, fallback, __arraycount(fallback));
168d99bda47Sriastradh 	} else {
169d99bda47Sriastradh 		(void)strlcat(buf, "\n", sizeof(buf));
170d99bda47Sriastradh 		(void)write(STDERR_FILENO, buf, strlen(buf));
171d99bda47Sriastradh 	}
172d99bda47Sriastradh }
1736e96c4eaSriastradh 
1746e96c4eaSriastradh void
block_signals(sigset_t * old_sigmask)1756e96c4eaSriastradh block_signals(sigset_t *old_sigmask)
1766e96c4eaSriastradh {
1776e96c4eaSriastradh 	sigset_t block;
1786e96c4eaSriastradh 
1796e96c4eaSriastradh 	(void)sigfillset(&block);
1806e96c4eaSriastradh 	(void)sigprocmask(SIG_BLOCK, &block, old_sigmask);
1816e96c4eaSriastradh }
1826e96c4eaSriastradh 
1836e96c4eaSriastradh void
restore_sigmask(const sigset_t * sigmask)1846e96c4eaSriastradh restore_sigmask(const sigset_t *sigmask)
1856e96c4eaSriastradh {
1866e96c4eaSriastradh 
1876e96c4eaSriastradh 	(void)sigprocmask(SIG_SETMASK, sigmask, NULL);
1886e96c4eaSriastradh }
189