aboutsummaryrefslogtreecommitdiffstats
path: root/eprintf.c
blob: 7a885b30c5365313aa9c0102f399f44a288f904c (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
#include <errno.h>
#include <stdarg.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <stdint.h>

#include "eprintf.h"

char *pname;

/* eprintf: print error and exit, a trailing : in fmt prints strerror(errno) */
void eprintf(const char *fmt, ...)
{
	int errnum = errno;
	va_list va;

	if (progname() != NULL)
		fprintf(stderr, "%s: ", progname());

	va_start(va, fmt);
	vfprintf(stderr, fmt, va);
	va_end(va);

	if (fmt[0] != '\0' && fmt[strlen(fmt) - 1] == ':')
		fprintf(stderr, " %s", strerror(errnum));
	fputc('\n', stderr);

	exit(EXIT_FAILURE);
}

/* weprintf: print warning, a trailing : in fmt prints strerror(errno) */
void weprintf(const char *fmt, ...)
{
	int errnum = errno;
	va_list va;

	if (progname() != NULL)
		fprintf(stderr, "%s: ", progname());

	va_start(va, fmt);
	vfprintf(stderr, fmt, va);
	va_end(va);

	if (fmt[0] != '\0' && fmt[strlen(fmt) - 1] == ':')
		fprintf(stderr, " %s", strerror(errnum));
	fputc('\n', stderr);
}

/* estrdup: attempt to strdup a string or exit on error */
char *estrdup(const char *s)
{
	size_t len;
	char *dup;

	len = strlen(s);
	if (len == SIZE_MAX)
		eprintf("estrdup(\"%.20s\") failed: Too long", s);
	dup = malloc(len + 1);
	if (dup == NULL)
		eprintf("estrdup(\"%.20s\") failed:", s);
	strcpy(dup, s);

	return dup;
}

/* emalloc: attempt to allocate memory or exit on error */
void *emalloc(size_t n)
{
	void *p;

	p = malloc(n);
	if (p == NULL)
		eprintf("emalloc(%zu) failed:", n);

	return p;
}

/* erealloc: attempt to reallocate memory or exit on error */
void *erealloc(void *p, size_t n)
{
	void *r;

	r = realloc(p, n);
	if (r == NULL)
		eprintf("erealloc(%p, %zu) failed:", p, n);

	return r;
}

/* progname: retrieve stored name of program */
char *progname(void)
{
	return pname;
}

/* setprogname: set stored name of program */
void setprogname(const char *s)
{
	if (pname != NULL) {
		weprintf("setprogname(\"%.20s\") called more than once", s);
		return;
	}

	pname = estrdup(s);
}