tail(1): improve style consistency of previous change
[dragonfly.git] / usr.bin / uname / uname.c
1 /*-
2  * Copyright (c) 2002 Juli Mallett.
3  * Copyright (c) 1993
4  *      The Regents of the University of California.  All rights reserved.
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  * 3. Neither the name of the University nor the names of its contributors
15  *    may be used to endorse or promote products derived from this software
16  *    without specific prior written permission.
17  *
18  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
19  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
20  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
21  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
22  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
23  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
24  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
25  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
26  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
27  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
28  * SUCH DAMAGE.
29  *
30  * @(#) Copyright (c) 1993 The Regents of the University of California.  All rights reserved.
31  * @(#)uname.c  8.2 (Berkeley) 5/4/95
32  * $FreeBSD: src/usr.bin/uname/uname.c,v 1.4.6.2 2002/10/17 07:47:29 jmallett Exp $
33  */
34
35 #include <sys/param.h>
36 #include <sys/sysctl.h>
37 #include <sys/varsym.h>
38
39 #include <err.h>
40 #include <stdio.h>
41 #include <stdlib.h>
42 #include <unistd.h>
43 #include <string.h>
44
45 #define MFLAG   0x0001
46 #define NFLAG   0x0002
47 #define PFLAG   0x0004
48 #define RFLAG   0x0008
49 #define SFLAG   0x0010
50 #define VFLAG   0x0020
51 #define IFLAG   0x0040
52 #define GFLAG   0x0080
53 #define GFLAG2  0x0100
54
55 typedef void (*get_t)(void);
56 static get_t get_ident, get_machine, get_hostname, get_arch;
57 static get_t get_release, get_sysname, get_version, get_pkgabi;
58   
59 static void native_ident(void);
60 static void native_machine(void);
61 static void native_hostname(void);
62 static void native_arch(void);
63 static void native_release(void);
64 static void native_sysname(void);
65 static void native_version(void);
66 static void native_pkgabi(void);
67 static void print_uname(void);
68 static void setup_get(void);
69 static void usage(void) __dead2;
70
71 static char *ident, *machine, *hostname, *arch;
72 static char *release, *sysname, *version, *pkgabi;
73 static int space;
74 static u_int flags;
75
76 int
77 main(int argc, char *argv[])
78 {
79         int ch;
80
81         setup_get();
82
83         while ((ch = getopt(argc, argv, "aimnprsvP")) != -1) {
84                 switch(ch) {
85                 case 'a':
86                         flags |= (MFLAG | NFLAG | RFLAG | SFLAG | VFLAG);
87                         break;
88                 case 'i':
89                         flags |= IFLAG;
90                         break;
91                 case 'm':
92                         flags |= MFLAG;
93                         break;
94                 case 'n':
95                         flags |= NFLAG;
96                         break;
97                 case 'p':
98                         flags |= PFLAG;
99                         break;
100                 case 'r':
101                         flags |= RFLAG;
102                         break;
103                 case 's':
104                         flags |= SFLAG;
105                         break;
106                 case 'v':
107                         flags |= VFLAG;
108                         break;
109                 case 'P':
110                         if (flags & GFLAG)      /* don't adjust odd numbers */
111                                 flags |= GFLAG2;
112                         flags |= GFLAG;
113                         break;
114                 case '?':
115                 default:
116                         usage();
117                 }
118         }
119
120         argc -= optind;
121         argv += optind;
122
123         if (argc)
124                 usage();
125
126         if (!flags)
127                 flags |= SFLAG;
128
129         print_uname();
130         exit(0);
131 }
132
133 /*
134  * Overrides.
135  *
136  * UNAME_x env variables have the highest priority
137  * UNAME_x varsyms have the next highest priority
138  * values retrieved from sysctls have the lowest priority
139  */
140 static
141 void
142 CHECK_ENV(const char *envname, get_t *getp, get_t nativep, char **varp)
143 {
144         char buf[MAXVARSYM_DATA];
145
146         if ((*varp = getenv(envname)) == NULL) {
147                 if (varsym_get(VARSYM_ALL_MASK, envname,
148                                buf, sizeof(buf)) < 0) {
149                         *getp = nativep;
150                         return;
151                 }
152                 *varp = strdup(buf);
153         }
154         *getp = NULL;
155 }
156
157 static void
158 setup_get(void)
159 {
160         CHECK_ENV("UNAME_s", &get_sysname, native_sysname, &sysname);
161         CHECK_ENV("UNAME_n", &get_hostname, native_hostname, &hostname);
162         CHECK_ENV("UNAME_r", &get_release, native_release, &release);
163         CHECK_ENV("UNAME_v", &get_version, native_version, &version);
164         CHECK_ENV("UNAME_m", &get_machine, native_machine, &machine);
165         CHECK_ENV("UNAME_p", &get_arch, native_arch, &arch);
166         CHECK_ENV("UNAME_i", &get_ident, native_ident, &ident);
167         CHECK_ENV("UNAME_G", &get_pkgabi, native_pkgabi, &pkgabi);
168 }
169
170 #define PRINT_FLAG(flags,flag,var)              \
171         if ((flags & flag) == flag) {           \
172                 if (space)                      \
173                         printf(" ");            \
174                 else                            \
175                         space++;                \
176                 if (get_##var != NULL)          \
177                         (*get_##var)();         \
178                 printf("%s", var);              \
179         }
180
181 static void
182 print_uname(void)
183 {
184         PRINT_FLAG(flags, SFLAG, sysname);
185         PRINT_FLAG(flags, NFLAG, hostname);
186         PRINT_FLAG(flags, RFLAG, release);
187         PRINT_FLAG(flags, VFLAG, version);
188         PRINT_FLAG(flags, MFLAG, machine);
189         PRINT_FLAG(flags, PFLAG, arch);
190         PRINT_FLAG(flags, IFLAG, ident);
191         PRINT_FLAG(flags, GFLAG, pkgabi);
192         printf("\n");
193 }
194
195 #define NATIVE_SYSCTL2_GET(var,mib0,mib1)       \
196 static void                                             \
197 native_##var(void)                              \
198 {                                               \
199         int mib[] = { (mib0), (mib1) };         \
200         size_t len;                             \
201         static char buf[1024];                  \
202         char **varp = &(var);                   \
203                                                 \
204         len = sizeof buf;                       \
205         if (sysctl(mib, NELEM(mib),             \
206            &buf, &len, NULL, 0) == -1)          \
207                 err(1, "sysctl");
208
209 #define NATIVE_SYSCTLNAME_GET(var,name)         \
210 static void                                             \
211 native_##var(void)                              \
212 {                                               \
213         size_t len;                             \
214         static char buf[1024];                  \
215         char **varp = &(var);                   \
216                                                 \
217         len = sizeof buf;                       \
218         if (sysctlbyname(name, &buf, &len, NULL,\
219             0) == -1)                           \
220                 err(1, "sysctlbyname");
221
222 #define NATIVE_SET                              \
223         *varp = buf;                            \
224         return;                                 \
225 }       struct __hack
226
227 #define NATIVE_BUFFER   (buf)
228 #define NATIVE_LENGTH   (len)
229
230 NATIVE_SYSCTL2_GET(sysname, CTL_KERN, KERN_OSTYPE) {
231 } NATIVE_SET;
232
233 NATIVE_SYSCTL2_GET(hostname, CTL_KERN, KERN_HOSTNAME) {
234 } NATIVE_SET;
235
236 NATIVE_SYSCTL2_GET(release, CTL_KERN, KERN_OSRELEASE) {
237 } NATIVE_SET;
238
239 NATIVE_SYSCTL2_GET(version, CTL_KERN, KERN_VERSION) {
240         size_t n;
241         char *p;
242
243         p = NATIVE_BUFFER;
244         n = NATIVE_LENGTH;
245         for (; n--; ++p)
246                 if (*p == '\n' || *p == '\t')
247                         *p = ' ';
248 } NATIVE_SET;
249
250 NATIVE_SYSCTL2_GET(machine, CTL_HW, HW_MACHINE) {
251 } NATIVE_SET;
252
253 NATIVE_SYSCTL2_GET(arch, CTL_HW, HW_MACHINE_ARCH) {
254 } NATIVE_SET;
255
256 NATIVE_SYSCTLNAME_GET(ident, "kern.ident") {
257 } NATIVE_SET;
258
259 static void                                             \
260 native_pkgabi(void)                             \
261 {
262         char osrel[64];
263         char mach[64];
264         size_t len;
265         double d;
266
267         len = sizeof(osrel);
268         if (sysctlbyname("kern.osrelease", osrel, &len, NULL, 0) == -1)
269                 err(1, "sysctlbyname");
270         len = sizeof(mach);
271         if (sysctlbyname("hw.machine", mach, &len, NULL, 0) == -1)
272                 err(1, "sysctlbyname");
273
274         /*
275          * Current convention is to adjust odd release numbers to even.
276          */
277         d = strtod(osrel, NULL);
278         if ((flags & GFLAG2) == 0) {
279                 if ((int)(d * 10) & 1)
280                         d = d + 0.1;    /* force to nearest even release */
281         }
282
283         /*
284          * pkgng expects the ABI in a different form
285          */
286         if (strcmp(mach, "x86_64") == 0)
287                 snprintf(mach, sizeof(mach), "x86:64");
288         else if (strcmp(mach, "i386") == 0)
289                 snprintf(mach, sizeof(mach), "x86:32");
290
291         asprintf(&pkgabi, "dragonfly:%3.1f:%s", d, mach);
292 }
293
294 static void
295 usage(void)
296 {
297         fprintf(stderr, "usage: uname [-aimnprsv]\n");
298         exit(1);
299 }