xref: /freebsd-src/contrib/bsnmp/lib/support.c (revision e1d581b289848ffa97c452756dc52d45efb68a01)
1*896052c1SHartmut Brandt /*
2*896052c1SHartmut Brandt  * Copyright (C) 2004
3*896052c1SHartmut Brandt  * 	Hartmut Brandt.
4*896052c1SHartmut Brandt  * 	All rights reserved.
5*896052c1SHartmut Brandt  *
6*896052c1SHartmut Brandt  * Author: Harti Brandt <harti@freebsd.org>
7*896052c1SHartmut Brandt  *
8*896052c1SHartmut Brandt  * Redistribution and use in source and binary forms, with or without
9*896052c1SHartmut Brandt  * modification, are permitted provided that the following conditions
10*896052c1SHartmut Brandt  * are met:
11*896052c1SHartmut Brandt  * 1. Redistributions of source code must retain the above copyright
12*896052c1SHartmut Brandt  *    notice, this list of conditions and the following disclaimer.
13*896052c1SHartmut Brandt  * 2. Redistributions in binary form must reproduce the above copyright
14*896052c1SHartmut Brandt  *    notice, this list of conditions and the following disclaimer in the
15*896052c1SHartmut Brandt  *    documentation and/or other materials provided with the distribution.
16*896052c1SHartmut Brandt  *
17*896052c1SHartmut Brandt  * THIS SOFTWARE IS PROVIDED BY AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18*896052c1SHartmut Brandt  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19*896052c1SHartmut Brandt  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20*896052c1SHartmut Brandt  * ARE DISCLAIMED.  IN NO EVENT SHALL AUTHOR OR CONTRIBUTORS BE LIABLE
21*896052c1SHartmut Brandt  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22*896052c1SHartmut Brandt  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23*896052c1SHartmut Brandt  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24*896052c1SHartmut Brandt  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25*896052c1SHartmut Brandt  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26*896052c1SHartmut Brandt  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27*896052c1SHartmut Brandt  * SUCH DAMAGE.
28*896052c1SHartmut Brandt  *
29*896052c1SHartmut Brandt  * $Begemot: bsnmp/lib/support.c,v 1.1 2004/08/06 08:47:58 brandt Exp $
30*896052c1SHartmut Brandt  *
31*896052c1SHartmut Brandt  * Functions that are missing on certain systems.
32*896052c1SHartmut Brandt  */
33*896052c1SHartmut Brandt #include <stdio.h>
34*896052c1SHartmut Brandt #include <stdlib.h>
35*896052c1SHartmut Brandt #include <stdarg.h>
36*896052c1SHartmut Brandt #include <errno.h>
37*896052c1SHartmut Brandt #include <string.h>
38*896052c1SHartmut Brandt #include "support.h"
39*896052c1SHartmut Brandt 
40*896052c1SHartmut Brandt #ifndef HAVE_ERR_H
41*896052c1SHartmut Brandt 
42*896052c1SHartmut Brandt void
warnx(const char * fmt,...)43*896052c1SHartmut Brandt warnx(const char *fmt, ...)
44*896052c1SHartmut Brandt {
45*896052c1SHartmut Brandt 	va_list ap;
46*896052c1SHartmut Brandt 
47*896052c1SHartmut Brandt 	va_start(ap, fmt);
48*896052c1SHartmut Brandt 	fprintf(stderr, "warning: ");
49*896052c1SHartmut Brandt 	vfprintf(stderr, fmt, ap);
50*896052c1SHartmut Brandt 	fprintf(stderr, "\n");
51*896052c1SHartmut Brandt 	va_end(ap);
52*896052c1SHartmut Brandt }
53*896052c1SHartmut Brandt 
54*896052c1SHartmut Brandt void
warn(const char * fmt,...)55*896052c1SHartmut Brandt warn(const char *fmt, ...)
56*896052c1SHartmut Brandt {
57*896052c1SHartmut Brandt 	va_list ap;
58*896052c1SHartmut Brandt 	int e = errno;
59*896052c1SHartmut Brandt 
60*896052c1SHartmut Brandt 	va_start(ap, fmt);
61*896052c1SHartmut Brandt 	fprintf(stderr, "warning: ");
62*896052c1SHartmut Brandt 	vfprintf(stderr, fmt, ap);
63*896052c1SHartmut Brandt 	fprintf(stderr, ": %s\n", strerror(e));
64*896052c1SHartmut Brandt 	va_end(ap);
65*896052c1SHartmut Brandt }
66*896052c1SHartmut Brandt 
67*896052c1SHartmut Brandt void
errx(int code,const char * fmt,...)68*896052c1SHartmut Brandt errx(int code, const char *fmt, ...)
69*896052c1SHartmut Brandt {
70*896052c1SHartmut Brandt 	va_list ap;
71*896052c1SHartmut Brandt 
72*896052c1SHartmut Brandt 	va_start(ap, fmt);
73*896052c1SHartmut Brandt 	fprintf(stderr, "error: ");
74*896052c1SHartmut Brandt 	vfprintf(stderr, fmt, ap);
75*896052c1SHartmut Brandt 	fprintf(stderr, "\n");
76*896052c1SHartmut Brandt 	va_end(ap);
77*896052c1SHartmut Brandt 	exit(code);
78*896052c1SHartmut Brandt }
79*896052c1SHartmut Brandt 
80*896052c1SHartmut Brandt void
err(int code,const char * fmt,...)81*896052c1SHartmut Brandt err(int code, const char *fmt, ...)
82*896052c1SHartmut Brandt {
83*896052c1SHartmut Brandt 	va_list ap;
84*896052c1SHartmut Brandt 	int e = errno;
85*896052c1SHartmut Brandt 
86*896052c1SHartmut Brandt 	va_start(ap, fmt);
87*896052c1SHartmut Brandt 	fprintf(stderr, "error: ");
88*896052c1SHartmut Brandt 	vfprintf(stderr, fmt, ap);
89*896052c1SHartmut Brandt 	fprintf(stderr, ": %s\n", strerror(e));
90*896052c1SHartmut Brandt 	va_end(ap);
91*896052c1SHartmut Brandt 	exit(code);
92*896052c1SHartmut Brandt }
93*896052c1SHartmut Brandt 
94*896052c1SHartmut Brandt #endif
95*896052c1SHartmut Brandt 
96*896052c1SHartmut Brandt #ifndef HAVE_STRLCPY
97*896052c1SHartmut Brandt 
98*896052c1SHartmut Brandt size_t
strlcpy(char * dst,const char * src,size_t len)99*896052c1SHartmut Brandt strlcpy(char *dst, const char *src, size_t len)
100*896052c1SHartmut Brandt {
101*896052c1SHartmut Brandt 	size_t ret = strlen(dst);
102*896052c1SHartmut Brandt 
103*896052c1SHartmut Brandt 	while (len > 1) {
104*896052c1SHartmut Brandt 		*dst++ = *src++;
105*896052c1SHartmut Brandt 		len--;
106*896052c1SHartmut Brandt 	}
107*896052c1SHartmut Brandt 	if (len > 0)
108*896052c1SHartmut Brandt 		*dst = '\0';
109*896052c1SHartmut Brandt 	return (ret);
110*896052c1SHartmut Brandt }
111*896052c1SHartmut Brandt 
112*896052c1SHartmut Brandt #endif
113*896052c1SHartmut Brandt 
114*896052c1SHartmut Brandt #ifndef HAVE_GETADDRINFO
115*896052c1SHartmut Brandt 
116*896052c1SHartmut Brandt #include <sys/types.h>
117*896052c1SHartmut Brandt #include <sys/socket.h>
118*896052c1SHartmut Brandt #include <netinet/in.h>
119*896052c1SHartmut Brandt #include <netdb.h>
120*896052c1SHartmut Brandt 
121*896052c1SHartmut Brandt extern int h_nerr;
122*896052c1SHartmut Brandt extern int h_errno;
123*896052c1SHartmut Brandt extern const char *h_errlist[];
124*896052c1SHartmut Brandt 
125*896052c1SHartmut Brandt /*
126*896052c1SHartmut Brandt  * VERY poor man's implementation
127*896052c1SHartmut Brandt  */
128*896052c1SHartmut Brandt int
getaddrinfo(const char * host,const char * port,const struct addrinfo * hints,struct addrinfo ** res)129*896052c1SHartmut Brandt getaddrinfo(const char *host, const char *port, const struct addrinfo *hints,
130*896052c1SHartmut Brandt     struct addrinfo **res)
131*896052c1SHartmut Brandt {
132*896052c1SHartmut Brandt 	struct hostent *hent;
133*896052c1SHartmut Brandt 	struct sockaddr_in *s;
134*896052c1SHartmut Brandt 	struct servent *sent;
135*896052c1SHartmut Brandt 
136*896052c1SHartmut Brandt 	if ((hent = gethostbyname(host)) == NULL)
137*896052c1SHartmut Brandt 		return (h_errno);
138*896052c1SHartmut Brandt 	if (hent->h_addrtype != hints->ai_family)
139*896052c1SHartmut Brandt 		return (HOST_NOT_FOUND);
140*896052c1SHartmut Brandt 	if (hent->h_addrtype != AF_INET)
141*896052c1SHartmut Brandt 		return (HOST_NOT_FOUND);
142*896052c1SHartmut Brandt 
143*896052c1SHartmut Brandt 	if ((*res = malloc(sizeof(**res))) == NULL)
144*896052c1SHartmut Brandt 		return (HOST_NOT_FOUND);
145*896052c1SHartmut Brandt 
146*896052c1SHartmut Brandt 	(*res)->ai_flags = hints->ai_flags;
147*896052c1SHartmut Brandt 	(*res)->ai_family = hints->ai_family;
148*896052c1SHartmut Brandt 	(*res)->ai_socktype = hints->ai_socktype;
149*896052c1SHartmut Brandt 	(*res)->ai_protocol = hints->ai_protocol;
150*896052c1SHartmut Brandt 	(*res)->ai_next = NULL;
151*896052c1SHartmut Brandt 
152*896052c1SHartmut Brandt 	if (((*res)->ai_addr = malloc(sizeof(struct sockaddr_in))) == NULL) {
153*896052c1SHartmut Brandt 		freeaddrinfo(*res);
154*896052c1SHartmut Brandt 		return (HOST_NOT_FOUND);
155*896052c1SHartmut Brandt 	}
156*896052c1SHartmut Brandt 	(*res)->ai_addrlen = sizeof(struct sockaddr_in);
157*896052c1SHartmut Brandt 	s = (struct sockaddr_in *)(*res)->ai_addr;
158*896052c1SHartmut Brandt 	s->sin_family = hints->ai_family;
159*896052c1SHartmut Brandt 	s->sin_len = sizeof(*s);
160*896052c1SHartmut Brandt 	memcpy(&s->sin_addr, hent->h_addr, 4);
161*896052c1SHartmut Brandt 
162*896052c1SHartmut Brandt 	if ((sent = getservbyname(port, NULL)) == NULL) {
163*896052c1SHartmut Brandt 		freeaddrinfo(*res);
164*896052c1SHartmut Brandt 		return (HOST_NOT_FOUND);
165*896052c1SHartmut Brandt 	}
166*896052c1SHartmut Brandt 	s->sin_port = sent->s_port;
167*896052c1SHartmut Brandt 
168*896052c1SHartmut Brandt 	return (0);
169*896052c1SHartmut Brandt }
170*896052c1SHartmut Brandt 
171*896052c1SHartmut Brandt const char *
gai_strerror(int e)172*896052c1SHartmut Brandt gai_strerror(int e)
173*896052c1SHartmut Brandt {
174*896052c1SHartmut Brandt 
175*896052c1SHartmut Brandt 	if (e < 0 || e >= h_nerr)
176*896052c1SHartmut Brandt 		return ("unknown error");
177*896052c1SHartmut Brandt 	return (h_errlist[e]);
178*896052c1SHartmut Brandt }
179*896052c1SHartmut Brandt 
180*896052c1SHartmut Brandt void
freeaddrinfo(struct addrinfo * p)181*896052c1SHartmut Brandt freeaddrinfo(struct addrinfo *p)
182*896052c1SHartmut Brandt {
183*896052c1SHartmut Brandt 	struct addrinfo *next;
184*896052c1SHartmut Brandt 
185*896052c1SHartmut Brandt 	while (p != NULL) {
186*896052c1SHartmut Brandt 		next = p->ai_next;
187*896052c1SHartmut Brandt 		if (p->ai_addr != NULL)
188*896052c1SHartmut Brandt 			free(p->ai_addr);
189*896052c1SHartmut Brandt 		free(p);
190*896052c1SHartmut Brandt 		p = next;
191*896052c1SHartmut Brandt 	}
192*896052c1SHartmut Brandt }
193*896052c1SHartmut Brandt 
194*896052c1SHartmut Brandt #endif
195