blob: 69d031dcae377fddde7b25a0228aaf6afb095e11 [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)
56# define ATTRIBUTE_ALWAYS_INLINE __attribute__ ((always_inline)) inline
57# else
58# define ATTRIBUTE_ALWAYS_INLINE inline
59# endif
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +000060
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000061/* -fwhole-program makes all symbols local. The attribute externally_visible
62 forces a symbol global. */
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000063# if __GNUC_PREREQ (4,1)
Mike Frysingerf8855132006-03-28 02:35:56 +000064# define ATTRIBUTE_EXTERNALLY_VISIBLE __attribute__ ((__externally_visible__))
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000065# else
Mike Frysingerf8855132006-03-28 02:35:56 +000066# define ATTRIBUTE_EXTERNALLY_VISIBLE
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000067# endif /* GNUC >= 4.1 */
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +000068
69/* We use __extension__ in some places to suppress -pedantic warnings
70 about GCC extensions. This feature didn't work properly before
71 gcc 2.8. */
72#if !__GNUC_PREREQ (2,8)
73# ifndef __extension__
74# define __extension__
75# endif
76#endif
77
Bernhard Reutner-Fischerb5f50ea2006-09-12 13:27:55 +000078/* gcc-2.95 had no va_copy but only __va_copy. */
79#if !__GNUC_PREREQ (3,0)
80# include <stdarg.h>
81# if !defined va_copy && defined __va_copy
82# define va_copy(d,s) __va_copy((d),(s))
83# endif
84#endif
85
Rob Landley5cf7c2d2006-02-21 06:44:43 +000086/* ---- Endian Detection ------------------------------------ */
Rob Landley5cf7c2d2006-02-21 06:44:43 +000087
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +000088#if (defined __digital__ && defined __unix__)
89# include <sex.h>
90# define __BIG_ENDIAN__ (BYTE_ORDER == BIG_ENDIAN)
91# define __BYTE_ORDER BYTE_ORDER
Rob Landley15d20a02006-05-29 05:00:44 +000092#elif !defined __APPLE__
93# include <byteswap.h>
94# include <endian.h>
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +000095#endif
96
Rob Landley5cf7c2d2006-02-21 06:44:43 +000097#ifdef __BIG_ENDIAN__
Mike Frysingerf8855132006-03-28 02:35:56 +000098# define BB_BIG_ENDIAN 1
99# define BB_LITTLE_ENDIAN 0
Rob Landley5cf7c2d2006-02-21 06:44:43 +0000100#elif __BYTE_ORDER == __BIG_ENDIAN
Mike Frysingerf8855132006-03-28 02:35:56 +0000101# define BB_BIG_ENDIAN 1
102# define BB_LITTLE_ENDIAN 0
Bernhard Reutner-Fischered7bb622006-02-23 14:25:15 +0000103#else
Mike Frysingerf8855132006-03-28 02:35:56 +0000104# define BB_BIG_ENDIAN 0
105# define BB_LITTLE_ENDIAN 1
Rob Landley5cf7c2d2006-02-21 06:44:43 +0000106#endif
107
Rob Landleybba7f082006-05-29 05:51:12 +0000108#if BB_BIG_ENDIAN
Rob Landley752f0a62006-05-30 06:28:03 +0000109#define SWAP_BE16(x) (x)
110#define SWAP_BE32(x) (x)
111#define SWAP_BE64(x) (x)
Rob Landleybba7f082006-05-29 05:51:12 +0000112#define SWAP_LE16(x) bswap_16(x)
113#define SWAP_LE32(x) bswap_32(x)
114#define SWAP_LE64(x) bswap_64(x)
115#else
116#define SWAP_BE16(x) bswap_16(x)
117#define SWAP_BE32(x) bswap_32(x)
118#define SWAP_BE64(x) bswap_64(x)
Rob Landley752f0a62006-05-30 06:28:03 +0000119#define SWAP_LE16(x) (x)
120#define SWAP_LE32(x) (x)
121#define SWAP_LE64(x) (x)
Rob Landleybba7f082006-05-29 05:51:12 +0000122#endif
123
Rob Landleydae6aa22006-03-09 22:39:08 +0000124/* ---- Networking ------------------------------------------ */
125#ifndef __APPLE__
Mike Frysingerf8855132006-03-28 02:35:56 +0000126# include <arpa/inet.h>
Rob Landleydae6aa22006-03-09 22:39:08 +0000127#else
Mike Frysingerf8855132006-03-28 02:35:56 +0000128# include <netinet/in.h>
Rob Landleydae6aa22006-03-09 22:39:08 +0000129#endif
130
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000131#ifndef __socklen_t_defined
132typedef int socklen_t;
133#endif
134
135/* ---- Compiler dependent settings ------------------------- */
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000136#if (defined __digital__ && defined __unix__)
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000137# undef HAVE_MNTENT_H
138#else
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000139# define HAVE_MNTENT_H 1
140#endif /* ___digital__ && __unix__ */
141
Bernhard Reutner-Fischerbe862092007-03-19 15:15:06 +0000142/* linux/loop.h relies on __u64. Make sure we have that as a proper type
143 * until userspace is widely fixed. */
144#ifndef __GNUC__
145#if defined __INTEL_COMPILER
146__extension__ typedef __signed__ long long __s64;
147__extension__ typedef unsigned long long __u64;
148#endif /* __INTEL_COMPILER */
149#endif /* ifndef __GNUC__ */
150
Bernhard Reutner-Fischere2e56c72006-05-19 11:54:02 +0000151/*----- Kernel versioning ------------------------------------*/
Bernhard Reutner-Fischere2e56c72006-05-19 11:54:02 +0000152#define KERNEL_VERSION(a,b,c) (((a) << 16) + ((b) << 8) + (c))
Bernhard Reutner-Fischere2e56c72006-05-19 11:54:02 +0000153
Bernhard Reutner-Fischered7bb622006-02-23 14:25:15 +0000154/* ---- miscellaneous --------------------------------------- */
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000155
Mike Frysingerb16b5bb2006-06-06 06:00:20 +0000156#if defined(__GNU_LIBRARY__) && __GNU_LIBRARY__ < 5 && \
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000157 !defined(__dietlibc__) && \
158 !defined(_NEWLIB_VERSION) && \
159 !(defined __digital__ && defined __unix__)
Mike Frysingerb16b5bb2006-06-06 06:00:20 +0000160# error "Sorry, this libc version is not supported :("
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000161#endif
162
Rob Landley18958e92006-06-13 18:28:33 +0000163// Don't perpetuate e2fsck crap into the headers. Clean up e2fsck instead.
164
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000165#if defined __GLIBC__ || defined __UCLIBC__ \
166 || defined __dietlibc__ || defined _NEWLIB_VERSION
167#include <features.h>
168#define HAVE_FEATURES_H
169#include <stdint.h>
170#define HAVE_STDINT_H
171#else
172/* Largest integral types. */
173#if __BIG_ENDIAN__
174typedef long int intmax_t;
175typedef unsigned long int uintmax_t;
176#else
177__extension__
178typedef long long int intmax_t;
179__extension__
180typedef unsigned long long int uintmax_t;
181#endif
182#endif
183
Bernhard Reutner-Fischerc966ba42007-01-18 10:32:09 +0000184/* Size-saving "small" ints (arch-dependent) */
185#if defined(i386) || defined(__x86_64__) || defined(__mips__) || defined(__cris__)
186/* add other arches which benefit from this... */
187typedef signed char smallint;
188typedef unsigned char smalluint;
189#else
190/* for arches where byte accesses generate larger code: */
191typedef int smallint;
192typedef unsigned smalluint;
193#endif
194
Bernhard Reutner-Fischera8e2e182007-01-20 21:27:18 +0000195/* ISO C Standard: 7.16 Boolean type and values <stdbool.h> */
196#if (defined __digital__ && defined __unix__)
197/* old system without (proper) C99 support */
198#define bool smalluint
199#else
200/* modern system, so use it */
201#include <stdbool.h>
202#endif
203
204
Mike Frysinger88407592006-07-20 19:31:07 +0000205/* uclibc does not implement daemon() for no-mmu systems.
Bernhard Reutner-Fischer507cd752006-05-31 10:04:03 +0000206 * For 0.9.29 and svn, __ARCH_USE_MMU__ indicates no-mmu reliably.
207 * For earlier versions there is no reliable way to check if we are building
Rob Landley1fa4a942006-06-22 22:05:00 +0000208 * for a mmu-less system; the user should pass EXTRA_CFLAGS="-DBB_NOMMU"
Bernhard Reutner-Fischer507cd752006-05-31 10:04:03 +0000209 * on his own.
210 */
211#if defined __UCLIBC__ && __UCLIBC_MAJOR__ >= 0 && __UCLIBC_MINOR__ >= 9 && \
212 __UCLIBC_SUBLEVEL__ > 28 && !defined __ARCH_USE_MMU__
213#define BB_NOMMU
214#endif
215
Rob Landley18958e92006-06-13 18:28:33 +0000216/* Platforms that haven't got dprintf need to implement fdprintf() in
217 * libbb. This would require a platform.c. It's not going to be cleaned
218 * out of the tree, so stop saying it should be. */
Denis Vlasenko7cfecc42006-12-18 22:32:45 +0000219#if !defined(__dietlibc__)
220/* Needed for: glibc */
221/* Not needed for: dietlibc */
222/* Others: ?? (add as needed) */
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000223#define fdprintf dprintf
Denis Vlasenko7cfecc42006-12-18 22:32:45 +0000224#endif
225
226#if defined(__dietlibc__)
227static ATTRIBUTE_ALWAYS_INLINE char* strchrnul(const char *s, char c) {
228 while (*s && *s != c) ++s;
229 return (char*)s;
230}
Denis Vlasenkoc8e6e352006-12-18 22:10:24 +0000231#endif
Rob Landleyc020f5f2006-05-21 18:28:13 +0000232
Denis Vlasenkoa597aad2006-12-16 23:48:13 +0000233/* Don't use lchown with glibc older than 2.1.x ... uC-libc lacks it */
Mike Frysinger88407592006-07-20 19:31:07 +0000234#if (defined __GLIBC__ && __GLIBC__ <= 2 && __GLIBC_MINOR__ < 1) || \
235 defined __UC_LIBC__
236# define lchown chown
237#endif
238
Bernhard Reutner-Fischer4ed1f1d2006-05-26 13:34:25 +0000239/* THIS SHOULD BE CLEANED OUT OF THE TREE ENTIRELY */
240/* FIXME: fix tar.c! */
241#ifndef FNM_LEADING_DIR
242#define FNM_LEADING_DIR 0
243#endif
244
Bernhard Reutner-Fischere00fc162006-05-26 13:10:10 +0000245#if (defined __digital__ && defined __unix__)
Rob Landley18958e92006-06-13 18:28:33 +0000246#include <standards.h>
247#define HAVE_STANDARDS_H
248#include <inttypes.h>
249#define HAVE_INTTYPES_H
250#define PRIu32 "u"
Rob Landley8fba99f2006-05-27 22:08:01 +0000251
Rob Landley18958e92006-06-13 18:28:33 +0000252/* use legacy setpgrp(pidt_,pid_t) for now. move to platform.c */
Denis Vlasenkobf0a2012006-12-26 10:42:51 +0000253#define bb_setpgrp do { pid_t __me = getpid(); setpgrp(__me,__me); } while (0)
Rob Landley18958e92006-06-13 18:28:33 +0000254
Rob Landley8fba99f2006-05-27 22:08:01 +0000255#if !defined ADJ_OFFSET_SINGLESHOT && defined MOD_CLKA && defined MOD_OFFSET
256#define ADJ_OFFSET_SINGLESHOT (MOD_CLKA | MOD_OFFSET)
257#endif
258#if !defined ADJ_FREQUENCY && defined MOD_FREQUENCY
259#define ADJ_FREQUENCY MOD_FREQUENCY
260#endif
261#if !defined ADJ_TIMECONST && defined MOD_TIMECONST
262#define ADJ_TIMECONST MOD_TIMECONST
263#endif
264#if !defined ADJ_TICK && defined MOD_CLKB
265#define ADJ_TICK MOD_CLKB
266#endif
267
Rob Landley18958e92006-06-13 18:28:33 +0000268#else
269#define bb_setpgrp setpgrp()
270#endif
271
Rob Landley22d26fc2006-06-15 15:49:36 +0000272#if defined(__linux__)
273#include <sys/mount.h>
Rob Landleye3781b72006-08-08 01:39:49 +0000274// Make sure we have all the new mount flags we actually try to use.
275#ifndef MS_BIND
276#define MS_BIND (1<<12)
277#endif
278#ifndef MS_MOVE
279#define MS_MOVE (1<<13)
280#endif
281#ifndef MS_RECURSIVE
282#define MS_RECURSIVE (1<<14)
283#endif
284#ifndef MS_SILENT
285#define MS_SILENT (1<<15)
286#endif
287
288// The shared subtree stuff, which went in around 2.6.15
289#ifndef MS_UNBINDABLE
290#define MS_UNBINDABLE (1<<17)
291#endif
292#ifndef MS_PRIVATE
293#define MS_PRIVATE (1<<18)
294#endif
295#ifndef MS_SLAVE
296#define MS_SLAVE (1<<19)
297#endif
298#ifndef MS_SHARED
299#define MS_SHARED (1<<20)
300#endif
301
Denis Vlasenko9213a9e2006-09-17 16:28:10 +0000302
Rob Landley7077ea32006-06-29 19:00:12 +0000303#if !defined(BLKSSZGET)
304#define BLKSSZGET _IO(0x12, 104)
305#endif
Rob Landley22d26fc2006-06-15 15:49:36 +0000306#if !defined(BLKGETSIZE64)
Rob Landley18958e92006-06-13 18:28:33 +0000307#define BLKGETSIZE64 _IOR(0x12,114,size_t)
308#endif
Rob Landley22d26fc2006-06-15 15:49:36 +0000309#endif
Rob Landley18958e92006-06-13 18:28:33 +0000310
Bernhard Reutner-Fischer86f5c992006-01-22 22:55:11 +0000311#endif /* platform.h */