1 /* vi: set sw=4 ts=4: */
3 Copyright 2006, Bernhard Reutner-Fischer
5 Licensed under the GPL v2 or later, see the file LICENSE in this tarball.
8 #define BB_PLATFORM_H 1
10 /* Convenience macros to test the version of gcc. */
12 #if defined __GNUC__ && defined __GNUC_MINOR__
13 # define __GNUC_PREREQ(maj, min) \
14 ((__GNUC__ << 16) + __GNUC_MINOR__ >= ((maj) << 16) + (min))
16 # define __GNUC_PREREQ(maj, min) 0
19 /* __restrict is known in EGCS 1.2 and above. */
20 #if !__GNUC_PREREQ(2,92)
22 # define __restrict /* Ignore */
26 /* Define macros for some gcc attributes. This permits us to use the
27 macros freely, and know that they will come into play for the
28 version of gcc in which they are supported. */
30 #if !__GNUC_PREREQ(2,7)
31 # ifndef __attribute__
32 # define __attribute__(x)
37 #if defined(__STDC_VERSION__) && __STDC_VERSION__ > 199901L
40 # if __GNUC_PREREQ(2,7)
41 # define inline __inline__
48 # define __const const
51 #define UNUSED_PARAM __attribute__ ((__unused__))
52 #define NORETURN __attribute__ ((__noreturn__))
53 #define PACKED __attribute__ ((__packed__))
54 #define ALIGNED(m) __attribute__ ((__aligned__(m)))
55 /* __NO_INLINE__: some gcc's do not honor inlining! :( */
56 #if __GNUC_PREREQ(3,0) && !defined(__NO_INLINE__)
57 # define ALWAYS_INLINE __attribute__ ((always_inline)) inline
58 /* I've seen a toolchain where I needed __noinline__ instead of noinline */
59 # define NOINLINE __attribute__((__noinline__))
61 # define DEPRECATED __attribute__ ((__deprecated__))
62 # define UNUSED_PARAM_RESULT __attribute__ ((warn_unused_result))
64 # define DEPRECATED /* n/a */
65 # define UNUSED_PARAM_RESULT /* n/a */
68 # define ALWAYS_INLINE inline /* n/a */
69 # define NOINLINE /* n/a */
70 # define DEPRECATED /* n/a */
71 # define UNUSED_PARAM_RESULT /* n/a */
74 /* -fwhole-program makes all symbols local. The attribute externally_visible
75 forces a symbol global. */
76 #if __GNUC_PREREQ(4,1)
77 # define EXTERNALLY_VISIBLE __attribute__(( visibility("default") ))
78 //__attribute__ ((__externally_visible__))
80 # define EXTERNALLY_VISIBLE
83 /* We use __extension__ in some places to suppress -pedantic warnings
84 about GCC extensions. This feature didn't work properly before
86 #if !__GNUC_PREREQ(2,8)
87 # ifndef __extension__
88 # define __extension__
92 /* gcc-2.95 had no va_copy but only __va_copy. */
93 #if !__GNUC_PREREQ(3,0)
95 # if !defined va_copy && defined __va_copy
96 # define va_copy(d,s) __va_copy((d),(s))
100 /* FAST_FUNC is a qualifier which (possibly) makes function call faster
101 * and/or smaller by using modified ABI. It is usually only needed
102 * on non-static, busybox internal functions. Recent versions of gcc
103 * optimize statics automatically. FAST_FUNC on static is required
104 * only if you need to match a function pointer's type */
105 #if __GNUC_PREREQ(3,0) && defined(i386) /* || defined(__x86_64__)? */
106 /* stdcall makes callee to pop arguments from stack, not caller */
107 # define FAST_FUNC __attribute__((regparm(3),stdcall))
108 /* #elif ... - add your favorite arch today! */
113 /* Make all declarations hidden (-fvisibility flag only affects definitions) */
114 /* (don't include system headers after this until corresponding pop!) */
115 #if __GNUC_PREREQ(4,1)
116 # define PUSH_AND_SET_FUNCTION_VISIBILITY_TO_HIDDEN _Pragma("GCC visibility push(hidden)")
117 # define POP_SAVED_FUNCTION_VISIBILITY _Pragma("GCC visibility pop")
119 # define PUSH_AND_SET_FUNCTION_VISIBILITY_TO_HIDDEN
120 # define POP_SAVED_FUNCTION_VISIBILITY
123 /* ---- Endian Detection ------------------------------------ */
125 #if (defined __digital__ && defined __unix__)
127 # define __BIG_ENDIAN__ (BYTE_ORDER == BIG_ENDIAN)
128 # define __BYTE_ORDER BYTE_ORDER
129 #elif !defined __APPLE__
130 # include <byteswap.h>
134 #ifdef __BIG_ENDIAN__
135 # define BB_BIG_ENDIAN 1
136 # define BB_LITTLE_ENDIAN 0
137 #elif __BYTE_ORDER == __BIG_ENDIAN
138 # define BB_BIG_ENDIAN 1
139 # define BB_LITTLE_ENDIAN 0
141 # define BB_BIG_ENDIAN 0
142 # define BB_LITTLE_ENDIAN 1
145 /* SWAP_LEnn means "convert CPU<->little_endian by swapping bytes" */
147 #define SWAP_BE16(x) (x)
148 #define SWAP_BE32(x) (x)
149 #define SWAP_BE64(x) (x)
150 #define SWAP_LE16(x) bswap_16(x)
151 #define SWAP_LE32(x) bswap_32(x)
152 #define SWAP_LE64(x) bswap_64(x)
154 #define SWAP_BE16(x) bswap_16(x)
155 #define SWAP_BE32(x) bswap_32(x)
156 #define SWAP_BE64(x) bswap_64(x)
157 #define SWAP_LE16(x) (x)
158 #define SWAP_LE32(x) (x)
159 #define SWAP_LE64(x) (x)
162 /* ---- Unaligned access ------------------------------------ */
164 /* NB: unaligned parameter should be a pointer, aligned one -
165 * a lvalue. This makes it more likely to not swap them by mistake
167 #if defined(i386) || defined(__x86_64__)
168 #define move_from_unaligned16(v, u16p) ((v) = *(uint16_t*)(u16p))
169 #define move_from_unaligned32(v, u32p) ((v) = *(uint32_t*)(u32p))
170 #define move_to_unaligned32(u32p, v) (*(uint32_t*)(u32p) = (v))
171 /* #elif ... - add your favorite arch today! */
173 /* performs reasonably well (gcc usually inlines memcpy here) */
174 #define move_from_unaligned16(v, u16p) (memcpy(&(v), (u16p), 2))
175 #define move_from_unaligned32(v, u32p) (memcpy(&(v), (u32p), 4))
176 #define move_to_unaligned32(u32p, v) (memcpy((u32p), &(v), 4))
179 /* ---- Networking ------------------------------------------ */
182 # include <arpa/inet.h>
183 # ifndef __socklen_t_defined
184 typedef int socklen_t
;
187 # include <netinet/in.h>
190 /* ---- Compiler dependent settings ------------------------- */
192 #if (defined __digital__ && defined __unix__) || defined __APPLE__
193 # undef HAVE_MNTENT_H
194 # undef HAVE_SYS_STATFS_H
196 # define HAVE_MNTENT_H 1
197 # define HAVE_SYS_STATFS_H 1
198 #endif /* ___digital__ && __unix__ */
200 /* linux/loop.h relies on __u64. Make sure we have that as a proper type
201 * until userspace is widely fixed. */
202 #if (defined __INTEL_COMPILER && !defined __GNUC__) || \
203 (defined __GNUC__ && defined __STRICT_ANSI__)
204 __extension__
typedef long long __s64
;
205 __extension__
typedef unsigned long long __u64
;
208 /*----- Kernel versioning ------------------------------------*/
210 #define KERNEL_VERSION(a,b,c) (((a) << 16) + ((b) << 8) + (c))
212 /* ---- Miscellaneous --------------------------------------- */
214 #if defined(__GNU_LIBRARY__) && __GNU_LIBRARY__ < 5 && \
215 !defined(__dietlibc__) && \
216 !defined(_NEWLIB_VERSION) && \
217 !(defined __digital__ && defined __unix__)
218 # error "Sorry, this libc version is not supported :("
221 /* Don't perpetuate e2fsck crap into the headers. Clean up e2fsck instead. */
223 #if defined __GLIBC__ || defined __UCLIBC__ \
224 || defined __dietlibc__ || defined _NEWLIB_VERSION
225 #include <features.h>
226 #define HAVE_FEATURES_H
228 #define HAVE_STDINT_H
229 #elif !defined __APPLE__
230 /* Largest integral types. */
232 typedef long intmax_t;
233 typedef unsigned long uintmax_t;
236 typedef long long intmax_t;
238 typedef unsigned long long uintmax_t;
242 /* Size-saving "small" ints (arch-dependent) */
243 #if defined(i386) || defined(__x86_64__) || defined(__mips__) || defined(__cris__)
244 /* add other arches which benefit from this... */
245 typedef signed char smallint
;
246 typedef unsigned char smalluint
;
248 /* for arches where byte accesses generate larger code: */
249 typedef int smallint
;
250 typedef unsigned smalluint
;
253 /* ISO C Standard: 7.16 Boolean type and values <stdbool.h> */
254 #if (defined __digital__ && defined __unix__)
255 /* old system without (proper) C99 support */
256 #define bool smalluint
258 /* modern system, so use it */
262 /* Try to defeat gcc's alignment of "char message[]"-like data */
263 #if 1 /* if needed: !defined(arch1) && !defined(arch2) */
264 #define ALIGN1 __attribute__((aligned(1)))
265 #define ALIGN2 __attribute__((aligned(2)))
267 /* Arches which MUST have 2 or 4 byte alignment for everything are here */
273 /* uclibc does not implement daemon() for no-mmu systems.
274 * For 0.9.29 and svn, __ARCH_USE_MMU__ indicates no-mmu reliably.
275 * For earlier versions there is no reliable way to check if we are building
276 * for a mmu-less system.
278 #if ENABLE_NOMMU || \
279 (defined __UCLIBC__ && __UCLIBC_MAJOR__ >= 0 && __UCLIBC_MINOR__ >= 9 && \
280 __UCLIBC_SUBLEVEL__ > 28 && !defined __ARCH_USE_MMU__)
282 #define USE_FOR_NOMMU(...) __VA_ARGS__
283 #define USE_FOR_MMU(...)
286 #define USE_FOR_NOMMU(...)
287 #define USE_FOR_MMU(...) __VA_ARGS__
290 /* Platforms that haven't got dprintf need to implement fdprintf() in
291 * libbb. This would require a platform.c. It's not going to be cleaned
292 * out of the tree, so stop saying it should be. */
293 #if !defined(__dietlibc__)
294 /* Needed for: glibc */
295 /* Not needed for: dietlibc */
296 /* Others: ?? (add as needed) */
297 #define fdprintf dprintf
300 #if defined(__dietlibc__)
301 static ALWAYS_INLINE
char* strchrnul(const char *s
, char c
)
303 while (*s
&& *s
!= c
) ++s
;
308 /* Don't use lchown with glibc older than 2.1.x ... uClibc lacks it */
309 #if (defined __GLIBC__ && __GLIBC__ <= 2 && __GLIBC_MINOR__ < 1) || \
311 # define lchown chown
314 #if (defined __digital__ && defined __unix__)
316 # include <standards.h>
317 # define HAVE_STANDARDS_H
318 # include <inttypes.h>
319 # define HAVE_INTTYPES_H
321 /* use legacy setpgrp(pid_t,pid_t) for now. move to platform.c */
322 # define bb_setpgrp() do { pid_t __me = getpid(); setpgrp(__me,__me); } while (0)
324 # if !defined ADJ_OFFSET_SINGLESHOT && defined MOD_CLKA && defined MOD_OFFSET
325 # define ADJ_OFFSET_SINGLESHOT (MOD_CLKA | MOD_OFFSET)
327 # if !defined ADJ_FREQUENCY && defined MOD_FREQUENCY
328 # define ADJ_FREQUENCY MOD_FREQUENCY
330 # if !defined ADJ_TIMECONST && defined MOD_TIMECONST
331 # define ADJ_TIMECONST MOD_TIMECONST
333 # if !defined ADJ_TICK && defined MOD_CLKB
334 # define ADJ_TICK MOD_CLKB
337 #else /* !__digital__ */
339 # define bb_setpgrp() setpgrp()
343 #if defined(__linux__)
344 #include <sys/mount.h>
345 /* Make sure we have all the new mount flags we actually try to use. */
347 #define MS_BIND (1<<12)
350 #define MS_MOVE (1<<13)
353 #define MS_RECURSIVE (1<<14)
356 #define MS_SILENT (1<<15)
359 /* The shared subtree stuff, which went in around 2.6.15. */
360 #ifndef MS_UNBINDABLE
361 #define MS_UNBINDABLE (1<<17)
364 #define MS_PRIVATE (1<<18)
367 #define MS_SLAVE (1<<19)
370 #define MS_SHARED (1<<20)
373 #define MS_RELATIME (1 << 21)
376 #if !defined(BLKSSZGET)
377 #define BLKSSZGET _IO(0x12, 104)
379 #if !defined(BLKGETSIZE64)
380 #define BLKGETSIZE64 _IOR(0x12,114,size_t)