blob: 050338cf3d013185f17e903b35750395baabdba4 [file] [log] [blame]
Bernhard Reutner-Fischerb1629b12006-05-19 19:29:19 +00001/* vi: set sw=4 ts=4: */
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +00002/*
3 Copyright 2006, Bernhard Fischer
4
5 Licensed under the GPL v2 or later, see the file LICENSE in this tarball.
6*/
7#ifndef __PLATFORM_H
8#define __PLATFORM_H 1
9
10/* Convenience macros to test the version of gcc. */
11#undef __GNUC_PREREQ
12#if defined __GNUC__ && defined __GNUC_MINOR__
13# define __GNUC_PREREQ(maj, min) \
Denis Vlasenko9213a9e2006-09-17 16:28:10 +000014 ((__GNUC__ << 16) + __GNUC_MINOR__ >= ((maj) << 16) + (min))
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000015#else
16# define __GNUC_PREREQ(maj, min) 0
17#endif
18
19/* __restrict is known in EGCS 1.2 and above. */
20#if !__GNUC_PREREQ (2,92)
21# ifndef __restrict
22# define __restrict /* Ignore */
23# endif
24#endif
25
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. */
29
30#if !__GNUC_PREREQ (2,7)
31# ifndef __attribute__
32# define __attribute__(x)
33# endif
34#endif
35
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +000036#undef inline
Denis Vlasenkoa7189f02006-11-17 20:29:00 +000037#if defined(__STDC_VERSION__) && __STDC_VERSION__ > 199901L
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +000038/* it's a keyword */
39#else
40# if __GNUC_PREREQ (2,7)
41# define inline __inline__
42# else
43# define inline
44# endif
45#endif
46
47#ifndef __const
48# define __const const
49#endif
50
Mike Frysingerf8855132006-03-28 02:35:56 +000051# define ATTRIBUTE_UNUSED __attribute__ ((__unused__))
Mike Frysingerf8855132006-03-28 02:35:56 +000052# define ATTRIBUTE_NORETURN __attribute__ ((__noreturn__))
Mike Frysingerf8855132006-03-28 02:35:56 +000053# define ATTRIBUTE_PACKED __attribute__ ((__packed__))
Mike Frysingerf8855132006-03-28 02:35:56 +000054# define ATTRIBUTE_ALIGNED(m) __attribute__ ((__aligned__(m)))
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +000055# if __GNUC_PREREQ (3,0)
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +000056# define ALWAYS_INLINE __attribute__ ((always_inline)) inline
Denis Vlasenko77f1ec12007-10-13 03:36:03 +000057/* I've seen a toolchain where I needed __noinline__ instead of noinline */
58# define NOINLINE __attribute__((__noinline__))
Denis Vlasenko734e5eb2007-05-04 21:38:14 +000059# if !ENABLE_WERROR
60# define ATTRIBUTE_DEPRECATED __attribute__ ((__deprecated__))
Bernhard Reutner-Fischer9a337802007-06-21 10:39:20 +000061# define ATTRIBUTE_UNUSED_RESULT __attribute__ ((warn_unused_result))
Denis Vlasenko734e5eb2007-05-04 21:38:14 +000062# else
63# define ATTRIBUTE_DEPRECATED /* n/a */
Bernhard Reutner-Fischer9a337802007-06-21 10:39:20 +000064# define ATTRIBUTE_UNUSED_RESULT /* n/a */
Denis Vlasenko734e5eb2007-05-04 21:38:14 +000065# endif
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +000066# else
Denis Vlasenko77f1ec12007-10-13 03:36:03 +000067# define ALWAYS_INLINE inline /* n/a */
68# define NOINLINE /* n/a */
Bernhard Reutner-Fischer51f7ab62007-04-10 18:40:05 +000069# define ATTRIBUTE_DEPRECATED /* n/a */
Bernhard Reutner-Fischer9a337802007-06-21 10:39:20 +000070# define ATTRIBUTE_UNUSED_RESULT /* n/a */
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +000071# endif
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +000072
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000073/* -fwhole-program makes all symbols local. The attribute externally_visible
74 forces a symbol global. */
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000075# if __GNUC_PREREQ (4,1)
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +000076# define EXTERNALLY_VISIBLE __attribute__(( visibility("default") ));
77//__attribute__ ((__externally_visible__))
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000078# else
Denis Vlasenko9b49a5e2007-10-11 10:05:36 +000079# define EXTERNALLY_VISIBLE
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000080# endif /* GNUC >= 4.1 */
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000081
82/* We use __extension__ in some places to suppress -pedantic warnings
83 about GCC extensions. This feature didn't work properly before
84 gcc 2.8. */
85#if !__GNUC_PREREQ (2,8)
86# ifndef __extension__
87# define __extension__
88# endif
89#endif
90
Bernhard Reutner-Fischerb5f50ea2006-09-12 13:27:55 +000091/* gcc-2.95 had no va_copy but only __va_copy. */
92#if !__GNUC_PREREQ (3,0)
93# include <stdarg.h>
94# if !defined va_copy && defined __va_copy
95# define va_copy(d,s) __va_copy((d),(s))
96# endif
97#endif
98
Rob Landley5cf7c2d2006-02-21 06:44:43 +000099/* ---- Endian Detection ------------------------------------ */
Rob Landley5cf7c2d2006-02-21 06:44:43 +0000100
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000101#if (defined __digital__ && defined __unix__)
102# include <sex.h>
103# define __BIG_ENDIAN__ (BYTE_ORDER == BIG_ENDIAN)
104# define __BYTE_ORDER BYTE_ORDER
Rob Landley15d20a02006-05-29 05:00:44 +0000105#elif !defined __APPLE__
106# include <byteswap.h>
107# include <endian.h>
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000108#endif
109
Rob Landley5cf7c2d2006-02-21 06:44:43 +0000110#ifdef __BIG_ENDIAN__
Mike Frysingerf8855132006-03-28 02:35:56 +0000111# define BB_BIG_ENDIAN 1
112# define BB_LITTLE_ENDIAN 0
Rob Landley5cf7c2d2006-02-21 06:44:43 +0000113#elif __BYTE_ORDER == __BIG_ENDIAN
Mike Frysingerf8855132006-03-28 02:35:56 +0000114# define BB_BIG_ENDIAN 1
115# define BB_LITTLE_ENDIAN 0
Bernhard Reutner-Fischered7bb622006-02-23 14:25:15 +0000116#else
Mike Frysingerf8855132006-03-28 02:35:56 +0000117# define BB_BIG_ENDIAN 0
118# define BB_LITTLE_ENDIAN 1
Rob Landley5cf7c2d2006-02-21 06:44:43 +0000119#endif
120
Rob Landleybba7f082006-05-29 05:51:12 +0000121#if BB_BIG_ENDIAN
Rob Landley752f0a62006-05-30 06:28:03 +0000122#define SWAP_BE16(x) (x)
123#define SWAP_BE32(x) (x)
124#define SWAP_BE64(x) (x)
Rob Landleybba7f082006-05-29 05:51:12 +0000125#define SWAP_LE16(x) bswap_16(x)
126#define SWAP_LE32(x) bswap_32(x)
127#define SWAP_LE64(x) bswap_64(x)
128#else
129#define SWAP_BE16(x) bswap_16(x)
130#define SWAP_BE32(x) bswap_32(x)
131#define SWAP_BE64(x) bswap_64(x)
Rob Landley752f0a62006-05-30 06:28:03 +0000132#define SWAP_LE16(x) (x)
133#define SWAP_LE32(x) (x)
134#define SWAP_LE64(x) (x)
Rob Landleybba7f082006-05-29 05:51:12 +0000135#endif
136
Rob Landleydae6aa22006-03-09 22:39:08 +0000137/* ---- Networking ------------------------------------------ */
138#ifndef __APPLE__
Mike Frysingerf8855132006-03-28 02:35:56 +0000139# include <arpa/inet.h>
Rob Landleydae6aa22006-03-09 22:39:08 +0000140#else
Mike Frysingerf8855132006-03-28 02:35:56 +0000141# include <netinet/in.h>
Rob Landleydae6aa22006-03-09 22:39:08 +0000142#endif
143
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000144#ifndef __socklen_t_defined
145typedef int socklen_t;
146#endif
147
148/* ---- Compiler dependent settings ------------------------- */
Mike Frysinger22876c72008-02-07 22:10:07 +0000149#if (defined __digital__ && defined __unix__) || defined __APPLE__
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000150# undef HAVE_MNTENT_H
151#else
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000152# define HAVE_MNTENT_H 1
153#endif /* ___digital__ && __unix__ */
154
Bernhard Reutner-Fischerbe862092007-03-19 15:15:06 +0000155/* linux/loop.h relies on __u64. Make sure we have that as a proper type
156 * until userspace is widely fixed. */
Bernhard Reutner-Fischer8c69afd2008-01-29 10:33:34 +0000157#if (defined __INTEL_COMPILER && !defined __GNUC__) || \
158 (defined __GNUC__ && defined __STRICT_ANSI__)
Bernhard Reutner-Fischerbe862092007-03-19 15:15:06 +0000159__extension__ typedef __signed__ long long __s64;
160__extension__ typedef unsigned long long __u64;
Bernhard Reutner-Fischer8c69afd2008-01-29 10:33:34 +0000161#endif
Bernhard Reutner-Fischerbe862092007-03-19 15:15:06 +0000162
Bernhard Reutner-Fischere2e56c72006-05-19 11:54:02 +0000163/*----- Kernel versioning ------------------------------------*/
Bernhard Reutner-Fischere2e56c72006-05-19 11:54:02 +0000164#define KERNEL_VERSION(a,b,c) (((a) << 16) + ((b) << 8) + (c))
Bernhard Reutner-Fischere2e56c72006-05-19 11:54:02 +0000165
Bernhard Reutner-Fischered7bb622006-02-23 14:25:15 +0000166/* ---- miscellaneous --------------------------------------- */
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000167
Mike Frysingerb16b5bb2006-06-06 06:00:20 +0000168#if defined(__GNU_LIBRARY__) && __GNU_LIBRARY__ < 5 && \
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000169 !defined(__dietlibc__) && \
170 !defined(_NEWLIB_VERSION) && \
171 !(defined __digital__ && defined __unix__)
Mike Frysingerb16b5bb2006-06-06 06:00:20 +0000172# error "Sorry, this libc version is not supported :("
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000173#endif
174
Denis Vlasenko6ca409e2007-08-12 20:58:27 +0000175/* Don't perpetuate e2fsck crap into the headers. Clean up e2fsck instead. */
Rob Landley18958e92006-06-13 18:28:33 +0000176
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000177#if defined __GLIBC__ || defined __UCLIBC__ \
178 || defined __dietlibc__ || defined _NEWLIB_VERSION
179#include <features.h>
180#define HAVE_FEATURES_H
181#include <stdint.h>
182#define HAVE_STDINT_H
183#else
184/* Largest integral types. */
185#if __BIG_ENDIAN__
Denis Vlasenko87468852007-04-13 23:22:00 +0000186typedef long intmax_t;
187typedef unsigned long uintmax_t;
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000188#else
189__extension__
Denis Vlasenko87468852007-04-13 23:22:00 +0000190typedef long long intmax_t;
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000191__extension__
Denis Vlasenko87468852007-04-13 23:22:00 +0000192typedef unsigned long long uintmax_t;
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000193#endif
194#endif
195
Bernhard Reutner-Fischerc966ba42007-01-18 10:32:09 +0000196/* Size-saving "small" ints (arch-dependent) */
197#if defined(i386) || defined(__x86_64__) || defined(__mips__) || defined(__cris__)
198/* add other arches which benefit from this... */
199typedef signed char smallint;
200typedef unsigned char smalluint;
201#else
202/* for arches where byte accesses generate larger code: */
203typedef int smallint;
204typedef unsigned smalluint;
205#endif
206
Bernhard Reutner-Fischera8e2e182007-01-20 21:27:18 +0000207/* ISO C Standard: 7.16 Boolean type and values <stdbool.h> */
208#if (defined __digital__ && defined __unix__)
209/* old system without (proper) C99 support */
210#define bool smalluint
211#else
212/* modern system, so use it */
213#include <stdbool.h>
214#endif
215
Denis Vlasenko6ca409e2007-08-12 20:58:27 +0000216/* Try to defeat gcc's alignment of "char message[]"-like data */
217#if 1 /* if needed: !defined(arch1) && !defined(arch2) */
218#define ALIGN1 __attribute__((aligned(1)))
219#define ALIGN2 __attribute__((aligned(2)))
220#else
221/* Arches which MUST have 2 or 4 byte alignment for everything are here */
222#define ALIGN1
223#define ALIGN2
224#endif
225
Bernhard Reutner-Fischera8e2e182007-01-20 21:27:18 +0000226
Mike Frysinger88407592006-07-20 19:31:07 +0000227/* uclibc does not implement daemon() for no-mmu systems.
Bernhard Reutner-Fischer507cd752006-05-31 10:04:03 +0000228 * For 0.9.29 and svn, __ARCH_USE_MMU__ indicates no-mmu reliably.
229 * For earlier versions there is no reliable way to check if we are building
Denis Vlasenko9e589212008-01-24 01:33:42 +0000230 * for a mmu-less system.
Bernhard Reutner-Fischer507cd752006-05-31 10:04:03 +0000231 */
Denis Vlasenkod2c450c2008-01-08 20:32:12 +0000232#if ENABLE_NOMMU || \
233 (defined __UCLIBC__ && __UCLIBC_MAJOR__ >= 0 && __UCLIBC_MINOR__ >= 9 && \
234 __UCLIBC_SUBLEVEL__ > 28 && !defined __ARCH_USE_MMU__)
Denis Vlasenko473dae02007-04-11 07:04:23 +0000235#define BB_MMU 0
236#define BB_NOMMU 1
237#define USE_FOR_NOMMU(...) __VA_ARGS__
238#define USE_FOR_MMU(...)
239#else
240#define BB_MMU 1
241/* BB_NOMMU is not defined in this case! */
242#define USE_FOR_NOMMU(...)
243#define USE_FOR_MMU(...) __VA_ARGS__
Bernhard Reutner-Fischer507cd752006-05-31 10:04:03 +0000244#endif
245
Rob Landley18958e92006-06-13 18:28:33 +0000246/* Platforms that haven't got dprintf need to implement fdprintf() in
247 * libbb. This would require a platform.c. It's not going to be cleaned
248 * out of the tree, so stop saying it should be. */
Denis Vlasenko7cfecc42006-12-18 22:32:45 +0000249#if !defined(__dietlibc__)
250/* Needed for: glibc */
251/* Not needed for: dietlibc */
252/* Others: ?? (add as needed) */
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000253#define fdprintf dprintf
Denis Vlasenko7cfecc42006-12-18 22:32:45 +0000254#endif
255
256#if defined(__dietlibc__)
Denis Vlasenko3ad5d0c2007-06-12 20:54:54 +0000257static ALWAYS_INLINE char* strchrnul(const char *s, char c)
Denis Vlasenkoac678ec2007-04-16 22:32:04 +0000258{
Denis Vlasenko7cfecc42006-12-18 22:32:45 +0000259 while (*s && *s != c) ++s;
260 return (char*)s;
261}
Denis Vlasenkoc8e6e352006-12-18 22:10:24 +0000262#endif
Rob Landleyc020f5f2006-05-21 18:28:13 +0000263
Denis Vlasenko473dae02007-04-11 07:04:23 +0000264/* Don't use lchown with glibc older than 2.1.x ... uClibc lacks it */
Mike Frysinger88407592006-07-20 19:31:07 +0000265#if (defined __GLIBC__ && __GLIBC__ <= 2 && __GLIBC_MINOR__ < 1) || \
266 defined __UC_LIBC__
267# define lchown chown
268#endif
269
Bernhard Reutner-Fischer4ed1f1d2006-05-26 13:34:25 +0000270/* THIS SHOULD BE CLEANED OUT OF THE TREE ENTIRELY */
271/* FIXME: fix tar.c! */
272#ifndef FNM_LEADING_DIR
273#define FNM_LEADING_DIR 0
274#endif
275
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000276#if (defined __digital__ && defined __unix__)
Rob Landley18958e92006-06-13 18:28:33 +0000277#include <standards.h>
278#define HAVE_STANDARDS_H
279#include <inttypes.h>
280#define HAVE_INTTYPES_H
281#define PRIu32 "u"
Rob Landley8fba99f2006-05-27 22:08:01 +0000282
Denis Vlasenko0de3c552007-04-12 12:31:02 +0000283/* use legacy setpgrp(pid_t,pid_t) for now. move to platform.c */
284#define bb_setpgrp() do { pid_t __me = getpid(); setpgrp(__me,__me); } while (0)
Rob Landley18958e92006-06-13 18:28:33 +0000285
Rob Landley8fba99f2006-05-27 22:08:01 +0000286#if !defined ADJ_OFFSET_SINGLESHOT && defined MOD_CLKA && defined MOD_OFFSET
287#define ADJ_OFFSET_SINGLESHOT (MOD_CLKA | MOD_OFFSET)
288#endif
289#if !defined ADJ_FREQUENCY && defined MOD_FREQUENCY
290#define ADJ_FREQUENCY MOD_FREQUENCY
291#endif
292#if !defined ADJ_TIMECONST && defined MOD_TIMECONST
293#define ADJ_TIMECONST MOD_TIMECONST
294#endif
295#if !defined ADJ_TICK && defined MOD_CLKB
296#define ADJ_TICK MOD_CLKB
297#endif
298
Rob Landley18958e92006-06-13 18:28:33 +0000299#else
Denis Vlasenko0de3c552007-04-12 12:31:02 +0000300#define bb_setpgrp() setpgrp()
Rob Landley18958e92006-06-13 18:28:33 +0000301#endif
302
Rob Landley22d26fc2006-06-15 15:49:36 +0000303#if defined(__linux__)
304#include <sys/mount.h>
Denis Vlasenko6ca409e2007-08-12 20:58:27 +0000305/* Make sure we have all the new mount flags we actually try to use. */
Rob Landleye3781b72006-08-08 01:39:49 +0000306#ifndef MS_BIND
307#define MS_BIND (1<<12)
308#endif
309#ifndef MS_MOVE
310#define MS_MOVE (1<<13)
311#endif
312#ifndef MS_RECURSIVE
313#define MS_RECURSIVE (1<<14)
314#endif
315#ifndef MS_SILENT
316#define MS_SILENT (1<<15)
317#endif
318
Denis Vlasenko6ca409e2007-08-12 20:58:27 +0000319/* The shared subtree stuff, which went in around 2.6.15. */
Rob Landleye3781b72006-08-08 01:39:49 +0000320#ifndef MS_UNBINDABLE
321#define MS_UNBINDABLE (1<<17)
322#endif
323#ifndef MS_PRIVATE
324#define MS_PRIVATE (1<<18)
325#endif
326#ifndef MS_SLAVE
327#define MS_SLAVE (1<<19)
328#endif
329#ifndef MS_SHARED
330#define MS_SHARED (1<<20)
331#endif
332
Denis Vlasenko9213a9e2006-09-17 16:28:10 +0000333
Rob Landley7077ea32006-06-29 19:00:12 +0000334#if !defined(BLKSSZGET)
335#define BLKSSZGET _IO(0x12, 104)
336#endif
Rob Landley22d26fc2006-06-15 15:49:36 +0000337#if !defined(BLKGETSIZE64)
Rob Landley18958e92006-06-13 18:28:33 +0000338#define BLKGETSIZE64 _IOR(0x12,114,size_t)
339#endif
Rob Landley22d26fc2006-06-15 15:49:36 +0000340#endif
Rob Landley18958e92006-06-13 18:28:33 +0000341
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +0000342#endif /* platform.h */