1 /* vi: set sw=4 ts=4: */
3 * Copyright 2006, Bernhard Reutner-Fischer
5 * Licensed under GPLv2 or later, see file LICENSE in this source tree.
8 #define BB_PLATFORM_H 1
10 /* Assume all these functions exist by default. Platforms where it is not
11 * true will #undef them below.
13 #define HAVE_FDPRINTF 1
14 #define HAVE_MEMRCHR 1
15 #define HAVE_MKDTEMP 1
17 #define HAVE_STRCASESTR 1
18 #define HAVE_STRCHRNUL 1
20 #define HAVE_STRSIGNAL 1
21 #define HAVE_VASPRINTF 1
23 /* Convenience macros to test the version of gcc. */
25 #if defined __GNUC__ && defined __GNUC_MINOR__
26 # define __GNUC_PREREQ(maj, min) \
27 ((__GNUC__ << 16) + __GNUC_MINOR__ >= ((maj) << 16) + (min))
29 # define __GNUC_PREREQ(maj, min) 0
32 /* __restrict is known in EGCS 1.2 and above. */
33 #if !__GNUC_PREREQ(2,92)
39 /* Define macros for some gcc attributes. This permits us to use the
40 macros freely, and know that they will come into play for the
41 version of gcc in which they are supported. */
43 #if !__GNUC_PREREQ(2,7)
44 # ifndef __attribute__
45 # define __attribute__(x)
50 #if defined(__STDC_VERSION__) && __STDC_VERSION__ > 199901L
52 #elif __GNUC_PREREQ(2,7)
53 # define inline __inline__
59 # define __const const
62 #define UNUSED_PARAM __attribute__ ((__unused__))
63 #define NORETURN __attribute__ ((__noreturn__))
64 /* "The malloc attribute is used to tell the compiler that a function
65 * may be treated as if any non-NULL pointer it returns cannot alias
66 * any other pointer valid when the function returns. This will often
67 * improve optimization. Standard functions with this property include
68 * malloc and calloc. realloc-like functions have this property as long
69 * as the old pointer is never referred to (including comparing it
70 * to the new pointer) after the function returns a non-NULL value."
72 #define RETURNS_MALLOC __attribute__ ((malloc))
73 #define PACKED __attribute__ ((__packed__))
74 #define ALIGNED(m) __attribute__ ((__aligned__(m)))
76 /* __NO_INLINE__: some gcc's do not honor inlining! :( */
77 #if __GNUC_PREREQ(3,0) && !defined(__NO_INLINE__)
78 # define ALWAYS_INLINE __attribute__ ((always_inline)) inline
79 /* I've seen a toolchain where I needed __noinline__ instead of noinline */
80 # define NOINLINE __attribute__((__noinline__))
82 # define DEPRECATED __attribute__ ((__deprecated__))
83 # define UNUSED_PARAM_RESULT __attribute__ ((warn_unused_result))
86 # define UNUSED_PARAM_RESULT
89 # define ALWAYS_INLINE inline
92 # define UNUSED_PARAM_RESULT
95 /* -fwhole-program makes all symbols local. The attribute externally_visible
96 forces a symbol global. */
97 #if __GNUC_PREREQ(4,1)
98 # define EXTERNALLY_VISIBLE __attribute__(( visibility("default") ))
99 //__attribute__ ((__externally_visible__))
101 # define EXTERNALLY_VISIBLE
104 /* At 4.4 gcc become much more anal about this, need to use "aliased" types */
105 #if __GNUC_PREREQ(4,4)
106 # define FIX_ALIASING __attribute__((__may_alias__))
108 # define FIX_ALIASING
111 /* We use __extension__ in some places to suppress -pedantic warnings
112 about GCC extensions. This feature didn't work properly before
114 #if !__GNUC_PREREQ(2,8)
115 # ifndef __extension__
116 # define __extension__
120 /* gcc-2.95 had no va_copy but only __va_copy. */
121 #if !__GNUC_PREREQ(3,0)
123 # if !defined va_copy && defined __va_copy
124 # define va_copy(d,s) __va_copy((d),(s))
128 /* FAST_FUNC is a qualifier which (possibly) makes function call faster
129 * and/or smaller by using modified ABI. It is usually only needed
130 * on non-static, busybox internal functions. Recent versions of gcc
131 * optimize statics automatically. FAST_FUNC on static is required
132 * only if you need to match a function pointer's type */
133 #if __GNUC_PREREQ(3,0) && defined(i386) /* || defined(__x86_64__)? */
134 /* stdcall makes callee to pop arguments from stack, not caller */
135 # define FAST_FUNC __attribute__((regparm(3),stdcall))
136 /* #elif ... - add your favorite arch today! */
141 /* Make all declarations hidden (-fvisibility flag only affects definitions) */
142 /* (don't include system headers after this until corresponding pop!) */
143 #if __GNUC_PREREQ(4,1)
144 # define PUSH_AND_SET_FUNCTION_VISIBILITY_TO_HIDDEN _Pragma("GCC visibility push(hidden)")
145 # define POP_SAVED_FUNCTION_VISIBILITY _Pragma("GCC visibility pop")
147 # define PUSH_AND_SET_FUNCTION_VISIBILITY_TO_HIDDEN
148 # define POP_SAVED_FUNCTION_VISIBILITY
151 /* ---- Endian Detection ------------------------------------ */
154 #if defined(__digital__) && defined(__unix__)
156 #elif defined(__FreeBSD__) || defined(__OpenBSD__) || defined(__NetBSD__) \
157 || defined(__APPLE__)
158 # include <sys/resource.h> /* rlimit */
159 # include <machine/endian.h>
160 # define bswap_64 __bswap64
161 # define bswap_32 __bswap32
162 # define bswap_16 __bswap16
164 # include <byteswap.h>
168 #if defined(__BYTE_ORDER) && __BYTE_ORDER == __BIG_ENDIAN
169 # define BB_BIG_ENDIAN 1
170 # define BB_LITTLE_ENDIAN 0
171 #elif defined(__BYTE_ORDER) && __BYTE_ORDER == __LITTLE_ENDIAN
172 # define BB_BIG_ENDIAN 0
173 # define BB_LITTLE_ENDIAN 1
174 #elif defined(_BYTE_ORDER) && _BYTE_ORDER == _BIG_ENDIAN
175 # define BB_BIG_ENDIAN 1
176 # define BB_LITTLE_ENDIAN 0
177 #elif defined(_BYTE_ORDER) && _BYTE_ORDER == _LITTLE_ENDIAN
178 # define BB_BIG_ENDIAN 0
179 # define BB_LITTLE_ENDIAN 1
180 #elif defined(BYTE_ORDER) && BYTE_ORDER == BIG_ENDIAN
181 # define BB_BIG_ENDIAN 1
182 # define BB_LITTLE_ENDIAN 0
183 #elif defined(BYTE_ORDER) && BYTE_ORDER == LITTLE_ENDIAN
184 # define BB_BIG_ENDIAN 0
185 # define BB_LITTLE_ENDIAN 1
186 #elif defined(__386__)
187 # define BB_BIG_ENDIAN 0
188 # define BB_LITTLE_ENDIAN 1
190 # error "Can't determine endianness"
193 #if ULONG_MAX > 0xffffffff
194 # define bb_bswap_64(x) bswap_64(x)
197 /* SWAP_LEnn means "convert CPU<->little_endian by swapping bytes" */
199 # define SWAP_BE16(x) (x)
200 # define SWAP_BE32(x) (x)
201 # define SWAP_BE64(x) (x)
202 # define SWAP_LE16(x) bswap_16(x)
203 # define SWAP_LE32(x) bswap_32(x)
204 # define SWAP_LE64(x) bb_bswap_64(x)
205 # define IF_BIG_ENDIAN(...) __VA_ARGS__
206 # define IF_LITTLE_ENDIAN(...)
208 # define SWAP_BE16(x) bswap_16(x)
209 # define SWAP_BE32(x) bswap_32(x)
210 # define SWAP_BE64(x) bb_bswap_64(x)
211 # define SWAP_LE16(x) (x)
212 # define SWAP_LE32(x) (x)
213 # define SWAP_LE64(x) (x)
214 # define IF_BIG_ENDIAN(...)
215 # define IF_LITTLE_ENDIAN(...) __VA_ARGS__
218 /* ---- Unaligned access ------------------------------------ */
220 /* NB: unaligned parameter should be a pointer, aligned one -
221 * a lvalue. This makes it more likely to not swap them by mistake
223 #if defined(i386) || defined(__x86_64__) || defined(__powerpc__)
225 typedef int bb__aliased_int FIX_ALIASING;
226 typedef uint16_t bb__aliased_uint16_t FIX_ALIASING;
227 typedef uint32_t bb__aliased_uint32_t FIX_ALIASING;
228 # define move_from_unaligned_int(v, intp) ((v) = *(bb__aliased_int*)(intp))
229 # define move_from_unaligned16(v, u16p) ((v) = *(bb__aliased_uint16_t*)(u16p))
230 # define move_from_unaligned32(v, u32p) ((v) = *(bb__aliased_uint32_t*)(u32p))
231 # define move_to_unaligned16(u16p, v) (*(bb__aliased_uint16_t*)(u16p) = (v))
232 # define move_to_unaligned32(u32p, v) (*(bb__aliased_uint32_t*)(u32p) = (v))
233 /* #elif ... - add your favorite arch today! */
235 /* performs reasonably well (gcc usually inlines memcpy here) */
236 # define move_from_unaligned_int(v, intp) (memcpy(&(v), (intp), sizeof(int)))
237 # define move_from_unaligned16(v, u16p) (memcpy(&(v), (u16p), 2))
238 # define move_from_unaligned32(v, u32p) (memcpy(&(v), (u32p), 4))
239 # define move_to_unaligned16(u16p, v) do { \
240 uint16_t __t = (v); \
241 memcpy((u16p), &__t, 4); \
243 # define move_to_unaligned32(u32p, v) do { \
244 uint32_t __t = (v); \
245 memcpy((u32p), &__t, 4); \
249 /* ---- Compiler dependent settings ------------------------- */
251 #if (defined __digital__ && defined __unix__) \
252 || defined __APPLE__ \
253 || defined __FreeBSD__ || defined __OpenBSD__ || defined __NetBSD__
254 # undef HAVE_MNTENT_H
255 # undef HAVE_SYS_STATFS_H
257 # define HAVE_MNTENT_H 1
258 # define HAVE_SYS_STATFS_H 1
261 /*----- Kernel versioning ------------------------------------*/
263 #define KERNEL_VERSION(a,b,c) (((a) << 16) + ((b) << 8) + (c))
265 /* ---- Miscellaneous --------------------------------------- */
267 #if defined __GLIBC__ || defined __UCLIBC__ \
268 || defined __dietlibc__ || defined _NEWLIB_VERSION
269 # include <features.h>
272 /* Size-saving "small" ints (arch-dependent) */
273 #if defined(i386) || defined(__x86_64__) || defined(__mips__) || defined(__cris__)
274 /* add other arches which benefit from this... */
275 typedef signed char smallint;
276 typedef unsigned char smalluint;
278 /* for arches where byte accesses generate larger code: */
279 typedef int smallint;
280 typedef unsigned smalluint;
283 /* ISO C Standard: 7.16 Boolean type and values <stdbool.h> */
284 #if (defined __digital__ && defined __unix__)
285 /* old system without (proper) C99 support */
286 # define bool smalluint
288 /* modern system, so use it */
289 # include <stdbool.h>
292 /* Try to defeat gcc's alignment of "char message[]"-like data */
293 #if 1 /* if needed: !defined(arch1) && !defined(arch2) */
294 # define ALIGN1 __attribute__((aligned(1)))
295 # define ALIGN2 __attribute__((aligned(2)))
296 # define ALIGN4 __attribute__((aligned(4)))
298 /* Arches which MUST have 2 or 4 byte alignment for everything are here */
305 /* uclibc does not implement daemon() for no-mmu systems.
306 * For 0.9.29 and svn, __ARCH_USE_MMU__ indicates no-mmu reliably.
307 * For earlier versions there is no reliable way to check if we are building
308 * for a mmu-less system.
310 #if ENABLE_NOMMU || \
311 (defined __UCLIBC__ && __UCLIBC_MAJOR__ >= 0 && __UCLIBC_MINOR__ >= 9 && \
312 __UCLIBC_SUBLEVEL__ > 28 && !defined __ARCH_USE_MMU__)
314 # define USE_FOR_NOMMU(...) __VA_ARGS__
315 # define USE_FOR_MMU(...)
318 # define USE_FOR_NOMMU(...)
319 # define USE_FOR_MMU(...) __VA_ARGS__
322 /* Don't use lchown with glibc older than 2.1.x */
323 #if defined(__GLIBC__) && __GLIBC__ <= 2 && __GLIBC_MINOR__ < 1
324 # define lchown chown
327 #if defined(__digital__) && defined(__unix__)
329 # include <standards.h>
330 # include <inttypes.h>
332 /* use legacy setpgrp(pid_t,pid_t) for now. move to platform.c */
333 # define bb_setpgrp() do { pid_t __me = getpid(); setpgrp(__me, __me); } while (0)
334 # if !defined ADJ_OFFSET_SINGLESHOT && defined MOD_CLKA && defined MOD_OFFSET
335 # define ADJ_OFFSET_SINGLESHOT (MOD_CLKA | MOD_OFFSET)
337 # if !defined ADJ_FREQUENCY && defined MOD_FREQUENCY
338 # define ADJ_FREQUENCY MOD_FREQUENCY
340 # if !defined ADJ_TIMECONST && defined MOD_TIMECONST
341 # define ADJ_TIMECONST MOD_TIMECONST
343 # if !defined ADJ_TICK && defined MOD_CLKB
344 # define ADJ_TICK MOD_CLKB
349 # define bb_setpgrp() setpgrp()
353 #if defined(__GLIBC__)
354 # define fdprintf dprintf
357 #if defined(__dietlibc__)
358 # undef HAVE_STRCHRNUL
361 #if defined(__WATCOMC__)
362 # undef HAVE_FDPRINTF
366 # undef HAVE_STRCASESTR
367 # undef HAVE_STRCHRNUL
369 # undef HAVE_STRSIGNAL
370 # undef HAVE_VASPRINTF
373 #if defined(__FreeBSD__)
374 # undef HAVE_STRCHRNUL
378 * Now, define prototypes for all the functions defined in platform.c
379 * These must come after all the HAVE_* macros are defined (or not)
382 #ifndef HAVE_FDPRINTF
383 extern int fdprintf(int fd, const char *format, ...);
387 extern void *memrchr(const void *s, int c, size_t n) FAST_FUNC;
391 extern char *mkdtemp(char *template) FAST_FUNC;
395 # define setbit(a, b) ((a)[(b) >> 3] |= 1 << ((b) & 7))
396 # define clrbit(a, b) ((a)[(b) >> 3] &= ~(1 << ((b) & 7)))
399 #ifndef HAVE_STRCASESTR
400 extern char *strcasestr(const char *s, const char *pattern) FAST_FUNC;
403 #ifndef HAVE_STRCHRNUL
404 extern char *strchrnul(const char *s, int c) FAST_FUNC;
408 extern char *strsep(char **stringp, const char *delim) FAST_FUNC;
411 #ifndef HAVE_STRSIGNAL
412 /* Not exactly the same: instead of "Stopped" it shows "STOP" etc */
413 # define strsignal(sig) get_signame(sig)
416 #ifndef HAVE_VASPRINTF
417 extern int vasprintf(char **string_ptr, const char *format, va_list p) FAST_FUNC;