1 /* misc.h - prototypes for misc functions */
3 * GRUB -- GRand Unified Bootloader
4 * Copyright (C) 2002,2003,2005,2006,2007,2008,2009,2010 Free Software Foundation, Inc.
6 * GRUB is free software: you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, either version 3 of the License, or
9 * (at your option) any later version.
11 * GRUB is distributed in the hope that it will be useful,
12 * but WITHOUT ANY WARRANTY; without even the implied warranty of
13 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 * GNU General Public License for more details.
16 * You should have received a copy of the GNU General Public License
17 * along with GRUB. If not, see <http://www.gnu.org/licenses/>.
20 #ifndef GRUB_MISC_HEADER
21 #define GRUB_MISC_HEADER 1
24 #include <grub/types.h>
25 #include <grub/symbol.h>
27 #include <grub/i18n.h>
28 #include <grub/compiler.h>
30 #define ALIGN_UP(addr, align) \
31 ((addr + (typeof (addr)) align - 1) & ~((typeof (addr)) align - 1))
32 #define ALIGN_UP_OVERHEAD(addr, align) ((-(addr)) & ((typeof (addr)) (align) - 1))
33 #define ALIGN_DOWN(addr, align) \
34 ((addr) & ~((typeof (addr)) align - 1))
35 #define ARRAY_SIZE(array) (sizeof (array) / sizeof (array[0]))
36 #define COMPILE_TIME_ASSERT(cond) switch (0) { case 1: case !(cond): ; }
38 #define grub_dprintf(condition, ...) grub_real_dprintf(GRUB_FILE, __LINE__, condition, __VA_ARGS__)
40 void *EXPORT_FUNC(grub_memmove
) (void *dest
, const void *src
, grub_size_t n
);
41 char *EXPORT_FUNC(grub_strcpy
) (char *dest
, const char *src
);
44 grub_strncpy (char *dest
, const char *src
, int c
)
48 while ((*p
++ = *src
++) != '\0' && --c
)
55 grub_stpcpy (char *dest
, const char *src
)
67 /* XXX: If grub_memmove is too slow, we must implement grub_memcpy. */
69 grub_memcpy (void *dest
, const void *src
, grub_size_t n
)
71 return grub_memmove (dest
, src
, n
);
74 #if defined (__APPLE__) && defined(__i386__) && !defined (GRUB_UTIL)
75 #define GRUB_BUILTIN_ATTR __attribute__ ((regparm(0)))
77 #define GRUB_BUILTIN_ATTR
80 #if defined(__x86_64__) && !defined (GRUB_UTIL)
81 #if defined (__MINGW32__) || defined (__CYGWIN__) || defined (__MINGW64__)
82 #define GRUB_ASM_ATTR __attribute__ ((sysv_abi))
88 /* Prototypes for aliases. */
90 int GRUB_BUILTIN_ATTR
EXPORT_FUNC(memcmp
) (const void *s1
, const void *s2
, grub_size_t n
);
91 void *GRUB_BUILTIN_ATTR
EXPORT_FUNC(memmove
) (void *dest
, const void *src
, grub_size_t n
);
92 void *GRUB_BUILTIN_ATTR
EXPORT_FUNC(memcpy
) (void *dest
, const void *src
, grub_size_t n
);
93 void *GRUB_BUILTIN_ATTR
EXPORT_FUNC(memset
) (void *s
, int c
, grub_size_t n
);
96 void GRUB_BUILTIN_ATTR
EXPORT_FUNC (__bzero
) (void *s
, grub_size_t n
);
101 int EXPORT_FUNC(grub_memcmp
) (const void *s1
, const void *s2
, grub_size_t n
);
102 int EXPORT_FUNC(grub_strcmp
) (const char *s1
, const char *s2
);
103 int EXPORT_FUNC(grub_strncmp
) (const char *s1
, const char *s2
, grub_size_t n
);
105 char *EXPORT_FUNC(grub_strchr
) (const char *s
, int c
);
106 char *EXPORT_FUNC(grub_strrchr
) (const char *s
, int c
);
107 int EXPORT_FUNC(grub_strword
) (const char *s
, const char *w
);
109 /* Copied from gnulib.
110 Written by Bruno Haible <bruno@clisp.org>, 2005. */
112 grub_strstr (const char *haystack
, const char *needle
)
114 /* Be careful not to look at the entire extent of haystack or needle
115 until needed. This is useful because of these two cases:
116 - haystack may be very long, and a match of needle found early,
117 - needle may be very long, and not even a short initial segment of
118 needle may be found in haystack. */
121 /* Speed up the following searches of needle by caching its first
127 if (*haystack
== '\0')
131 /* The first character matches. */
133 const char *rhaystack
= haystack
+ 1;
134 const char *rneedle
= needle
;
136 for (;; rhaystack
++, rneedle
++)
138 if (*rneedle
== '\0')
140 return (char *) haystack
;
141 if (*rhaystack
== '\0')
144 if (*rhaystack
!= *rneedle
)
145 /* Nothing in this round. */
152 return (char *) haystack
;
155 int EXPORT_FUNC(grub_isspace
) (int c
);
160 return (c
>= ' ' && c
<= '~');
166 return (c
>= 0x00 && c
<= 0x1F) || c
== 0x7F;
172 return (c
>= 'a' && c
<= 'z') || (c
>= 'A' && c
<= 'Z');
178 return (c
>= 'a' && c
<= 'z');
184 return (c
>= 'A' && c
<= 'Z');
190 return (c
>= '!' && c
<= '~');
196 return (c
>= '0' && c
<= '9');
200 grub_isxdigit (int c
)
202 return (c
>= '0' && c
<= '9') || (c
>= 'a' && c
<= 'f') || (c
>= 'A' && c
<= 'F');
208 return grub_isalpha (c
) || grub_isdigit (c
);
214 if (c
>= 'A' && c
<= 'Z')
215 return c
- 'A' + 'a';
223 if (c
>= 'a' && c
<= 'z')
224 return c
- 'a' + 'A';
230 grub_strcasecmp (const char *s1
, const char *s2
)
234 if (grub_tolower ((grub_uint8_t
) *s1
)
235 != grub_tolower ((grub_uint8_t
) *s2
))
242 return (int) grub_tolower ((grub_uint8_t
) *s1
)
243 - (int) grub_tolower ((grub_uint8_t
) *s2
);
247 grub_strncasecmp (const char *s1
, const char *s2
, grub_size_t n
)
252 while (*s1
&& *s2
&& --n
)
254 if (grub_tolower (*s1
) != grub_tolower (*s2
))
261 return (int) grub_tolower ((grub_uint8_t
) *s1
)
262 - (int) grub_tolower ((grub_uint8_t
) *s2
);
265 unsigned long EXPORT_FUNC(grub_strtoul
) (const char *str
, char **end
, int base
);
266 unsigned long long EXPORT_FUNC(grub_strtoull
) (const char *str
, char **end
, int base
);
269 grub_strtol (const char *str
, char **end
, int base
)
272 unsigned long long magnitude
;
274 while (*str
&& grub_isspace (*str
))
283 magnitude
= grub_strtoull (str
, end
, base
);
286 if (magnitude
> (unsigned long) GRUB_LONG_MAX
+ 1)
288 grub_error (GRUB_ERR_OUT_OF_RANGE
, N_("overflow is detected"));
289 return GRUB_LONG_MIN
;
291 return -((long) magnitude
);
295 if (magnitude
> GRUB_LONG_MAX
)
297 grub_error (GRUB_ERR_OUT_OF_RANGE
, N_("overflow is detected"));
298 return GRUB_LONG_MAX
;
300 return (long) magnitude
;
304 char *EXPORT_FUNC(grub_strdup
) (const char *s
) WARN_UNUSED_RESULT
;
305 char *EXPORT_FUNC(grub_strndup
) (const char *s
, grub_size_t n
) WARN_UNUSED_RESULT
;
306 void *EXPORT_FUNC(grub_memset
) (void *s
, int c
, grub_size_t n
);
307 grub_size_t
EXPORT_FUNC(grub_strlen
) (const char *s
) WARN_UNUSED_RESULT
;
308 int EXPORT_FUNC(grub_printf
) (const char *fmt
, ...) __attribute__ ((format (GNU_PRINTF
, 1, 2)));
309 int EXPORT_FUNC(grub_printf_
) (const char *fmt
, ...) __attribute__ ((format (GNU_PRINTF
, 1, 2)));
311 /* Replace all `ch' characters of `input' with `with' and copy the
312 result into `output'; return EOS address of `output'. */
314 grub_strchrsub (char *output
, const char *input
, char ch
, const char *with
)
320 grub_strcpy (output
, with
);
321 output
+= grub_strlen (with
);
325 *output
++ = *input
++;
331 extern void (*EXPORT_VAR (grub_xputs
)) (const char *str
);
334 grub_puts (const char *s
)
336 const char nl
[2] = "\n";
340 return 1; /* Cannot fail. */
343 int EXPORT_FUNC(grub_puts_
) (const char *s
);
344 void EXPORT_FUNC(grub_real_dprintf
) (const char *file
,
346 const char *condition
,
347 const char *fmt
, ...) __attribute__ ((format (GNU_PRINTF
, 4, 5)));
348 int EXPORT_FUNC(grub_vprintf
) (const char *fmt
, va_list args
);
349 int EXPORT_FUNC(grub_snprintf
) (char *str
, grub_size_t n
, const char *fmt
, ...)
350 __attribute__ ((format (GNU_PRINTF
, 3, 4)));
351 int EXPORT_FUNC(grub_vsnprintf
) (char *str
, grub_size_t n
, const char *fmt
,
353 char *EXPORT_FUNC(grub_xasprintf
) (const char *fmt
, ...)
354 __attribute__ ((format (GNU_PRINTF
, 1, 2))) WARN_UNUSED_RESULT
;
355 char *EXPORT_FUNC(grub_xvasprintf
) (const char *fmt
, va_list args
) WARN_UNUSED_RESULT
;
356 void EXPORT_FUNC(grub_exit
) (void) __attribute__ ((noreturn
));
357 grub_uint64_t
EXPORT_FUNC(grub_divmod64
) (grub_uint64_t n
,
361 #if (defined (__MINGW32__) || defined (__CYGWIN__)) && !defined(GRUB_UTIL)
362 void EXPORT_FUNC (__register_frame_info
) (void);
363 void EXPORT_FUNC (__deregister_frame_info
) (void);
364 void EXPORT_FUNC (___chkstk_ms
) (void);
365 void EXPORT_FUNC (__chkstk_ms
) (void);
368 /* Inline functions. */
371 grub_memchr (const void *p
, int c
, grub_size_t len
)
373 const char *s
= (const char *) p
;
374 const char *e
= s
+ len
;
384 static inline unsigned int
388 return (unsigned int) (-x
);
390 return (unsigned int) x
;
393 /* Reboot the machine. */
394 #if defined (GRUB_MACHINE_EMU) || defined (GRUB_MACHINE_QEMU_MIPS)
395 void EXPORT_FUNC(grub_reboot
) (void) __attribute__ ((noreturn
));
397 void grub_reboot (void) __attribute__ ((noreturn
));
400 #if defined (__clang__) && !defined (GRUB_UTIL)
401 void __attribute__ ((noreturn
)) EXPORT_FUNC (abort
) (void);
404 #ifdef GRUB_MACHINE_PCBIOS
405 /* Halt the system, using APM if possible. If NO_APM is true, don't
406 * use APM even if it is available. */
407 void grub_halt (int no_apm
) __attribute__ ((noreturn
));
408 #elif defined (__mips__) && !defined (GRUB_MACHINE_EMU)
409 void EXPORT_FUNC (grub_halt
) (void) __attribute__ ((noreturn
));
411 void grub_halt (void) __attribute__ ((noreturn
));
414 #ifdef GRUB_MACHINE_EMU
415 /* Flag to check if module loading is available. */
416 extern const int EXPORT_VAR(grub_no_modules
);
418 #define grub_no_modules 0
422 grub_error_save (struct grub_error_saved
*save
)
424 grub_memcpy (save
->errmsg
, grub_errmsg
, sizeof (save
->errmsg
));
425 save
->grub_errno
= grub_errno
;
426 grub_errno
= GRUB_ERR_NONE
;
430 grub_error_load (const struct grub_error_saved
*save
)
432 grub_memcpy (grub_errmsg
, save
->errmsg
, sizeof (grub_errmsg
));
433 grub_errno
= save
->grub_errno
;
438 #if defined (__arm__)
441 EXPORT_FUNC (__udivsi3
) (grub_uint32_t a
, grub_uint32_t b
);
444 EXPORT_FUNC (__umodsi3
) (grub_uint32_t a
, grub_uint32_t b
);
448 #if defined (__sparc__) || defined (__powerpc__)
450 EXPORT_FUNC (__ctzdi2
) (grub_uint64_t x
);
451 #define NEED_CTZDI2 1
454 #if defined (__mips__) || defined (__arm__)
456 EXPORT_FUNC (__ctzsi2
) (grub_uint32_t x
);
457 #define NEED_CTZSI2 1
462 EXPORT_FUNC (__aeabi_uidiv
) (grub_uint32_t a
, grub_uint32_t b
);
464 EXPORT_FUNC (__aeabi_uidivmod
) (grub_uint32_t a
, grub_uint32_t b
);
466 /* Needed for allowing modules to be compiled as thumb. */
468 EXPORT_FUNC (__muldi3
) (grub_uint64_t a
, grub_uint64_t b
);
470 EXPORT_FUNC (__aeabi_lmul
) (grub_uint64_t a
, grub_uint64_t b
);
474 #if defined (__ia64__)
477 EXPORT_FUNC (__udivdi3
) (grub_uint64_t a
, grub_uint64_t b
);
480 EXPORT_FUNC (__umoddi3
) (grub_uint64_t a
, grub_uint64_t b
);
484 #endif /* GRUB_UTIL */
488 struct grub_boot_time
490 struct grub_boot_time
*next
;
497 extern struct grub_boot_time
*EXPORT_VAR(grub_boot_time_head
);
499 void EXPORT_FUNC(grub_real_boot_time
) (const char *file
,
501 const char *fmt
, ...) __attribute__ ((format (GNU_PRINTF
, 3, 4)));
502 #define grub_boot_time(...) grub_real_boot_time(GRUB_FILE, __LINE__, __VA_ARGS__)
504 #define grub_boot_time(...)
507 #define grub_max(a, b) (((a) > (b)) ? (a) : (b))
508 #define grub_min(a, b) (((a) < (b)) ? (a) : (b))
510 #endif /* ! GRUB_MISC_HEADER */