|  | /* | 
|  | * Copyright (c) Artem Bityutskiy, 2007, 2008 | 
|  | * | 
|  | * This program is free software; you can redistribute it and/or modify | 
|  | * it under the terms of the GNU General Public License as published by | 
|  | * the Free Software Foundation; either version 2 of the License, or | 
|  | * (at your option) any later version. | 
|  | * | 
|  | * This program is distributed in the hope that it will be useful, | 
|  | * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
|  | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See | 
|  | * the GNU General Public License for more details. | 
|  | * | 
|  | * You should have received a copy of the GNU General Public License | 
|  | * along with this program; if not, write to the Free Software | 
|  | * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | 
|  | */ | 
|  |  | 
|  | #ifndef __MTD_UTILS_COMMON_H__ | 
|  | #define __MTD_UTILS_COMMON_H__ | 
|  |  | 
|  | #include <stdbool.h> | 
|  | #include <stdio.h> | 
|  | #include <stdlib.h> | 
|  | #include <ctype.h> | 
|  | #include <string.h> | 
|  | #include <fcntl.h> | 
|  | #include <errno.h> | 
|  | #include <features.h> | 
|  | #include <inttypes.h> | 
|  |  | 
|  | #ifndef PROGRAM_NAME | 
|  | # error "You must define PROGRAM_NAME before including this header" | 
|  | #endif | 
|  |  | 
|  | #ifdef __cplusplus | 
|  | extern "C" { | 
|  | #endif | 
|  |  | 
|  | #ifndef MIN	/* some C lib headers define this for us */ | 
|  | #define MIN(a, b) ((a) < (b) ? (a) : (b)) | 
|  | #endif | 
|  | #ifndef MAX | 
|  | #define MAX(a, b) ((a) > (b) ? (a) : (b)) | 
|  | #endif | 
|  | #define min(a, b) MIN(a, b) /* glue for linux kernel source */ | 
|  | #define ARRAY_SIZE(a) (sizeof(a) / sizeof((a)[0])) | 
|  |  | 
|  | #define ALIGN(x,a) __ALIGN_MASK(x,(typeof(x))(a)-1) | 
|  | #define __ALIGN_MASK(x,mask) (((x)+(mask))&~(mask)) | 
|  |  | 
|  | #define min_t(t,x,y) ({ \ | 
|  | typeof((x)) _x = (x); \ | 
|  | typeof((y)) _y = (y); \ | 
|  | (_x < _y) ? _x : _y; \ | 
|  | }) | 
|  |  | 
|  | #define max_t(t,x,y) ({ \ | 
|  | typeof((x)) _x = (x); \ | 
|  | typeof((y)) _y = (y); \ | 
|  | (_x > _y) ? _x : _y; \ | 
|  | }) | 
|  |  | 
|  | #ifndef O_CLOEXEC | 
|  | #define O_CLOEXEC 0 | 
|  | #endif | 
|  |  | 
|  | /* define a print format specifier for off_t */ | 
|  | #ifdef __USE_FILE_OFFSET64 | 
|  | #define PRIxoff_t PRIx64 | 
|  | #define PRIdoff_t PRId64 | 
|  | #else | 
|  | #define PRIxoff_t "l"PRIx32 | 
|  | #define PRIdoff_t "l"PRId32 | 
|  | #endif | 
|  |  | 
|  | /* Verbose messages */ | 
|  | #define bareverbose(verbose, fmt, ...) do {                        \ | 
|  | if (verbose)                                               \ | 
|  | printf(fmt, ##__VA_ARGS__);                        \ | 
|  | } while(0) | 
|  | #define verbose(verbose, fmt, ...) \ | 
|  | bareverbose(verbose, "%s: " fmt "\n", PROGRAM_NAME, ##__VA_ARGS__) | 
|  |  | 
|  | /* Normal messages */ | 
|  | #define normsg_cont(fmt, ...) do {                                 \ | 
|  | printf("%s: " fmt, PROGRAM_NAME, ##__VA_ARGS__);           \ | 
|  | } while(0) | 
|  | #define normsg(fmt, ...) do {                                      \ | 
|  | normsg_cont(fmt "\n", ##__VA_ARGS__);                      \ | 
|  | } while(0) | 
|  |  | 
|  | /* Error messages */ | 
|  | #define errmsg(fmt, ...)  ({                                                \ | 
|  | fprintf(stderr, "%s: error!: " fmt "\n", PROGRAM_NAME, ##__VA_ARGS__); \ | 
|  | -1;                                                                 \ | 
|  | }) | 
|  | #define errmsg_die(fmt, ...) do {                                           \ | 
|  | exit(errmsg(fmt, ##__VA_ARGS__));                                   \ | 
|  | } while(0) | 
|  |  | 
|  | /* System error messages */ | 
|  | #define sys_errmsg(fmt, ...)  ({                                            \ | 
|  | int _err = errno;                                                   \ | 
|  | errmsg(fmt, ##__VA_ARGS__);                                         \ | 
|  | fprintf(stderr, "%*serror %d (%s)\n", (int)sizeof(PROGRAM_NAME) + 1,\ | 
|  | "", _err, strerror(_err));                                  \ | 
|  | -1;                                                                 \ | 
|  | }) | 
|  | #define sys_errmsg_die(fmt, ...) do {                                       \ | 
|  | exit(sys_errmsg(fmt, ##__VA_ARGS__));                               \ | 
|  | } while(0) | 
|  |  | 
|  | /* Warnings */ | 
|  | #define warnmsg(fmt, ...) do {                                                \ | 
|  | fprintf(stderr, "%s: warning!: " fmt "\n", PROGRAM_NAME, ##__VA_ARGS__); \ | 
|  | } while(0) | 
|  |  | 
|  | /* uClibc versions before 0.9.34 and musl don't have rpmatch() */ | 
|  | #if defined(__UCLIBC__) && \ | 
|  | (__UCLIBC_MAJOR__ == 0 && \ | 
|  | (__UCLIBC_MINOR__ < 9 || \ | 
|  | (__UCLIBC_MINOR__ == 9 && __UCLIBC_SUBLEVEL__ < 34))) || \ | 
|  | !defined(__GLIBC__) | 
|  | #undef rpmatch | 
|  | #define rpmatch __rpmatch | 
|  | static inline int __rpmatch(const char *resp) | 
|  | { | 
|  | return (resp[0] == 'y' || resp[0] == 'Y') ? 1 : | 
|  | (resp[0] == 'n' || resp[0] == 'N') ? 0 : -1; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | /** | 
|  | * prompt the user for confirmation | 
|  | */ | 
|  | static inline bool prompt(const char *msg, bool def) | 
|  | { | 
|  | char *line = NULL; | 
|  | size_t len; | 
|  | bool ret = def; | 
|  |  | 
|  | do { | 
|  | normsg_cont("%s (%c/%c) ", msg, def ? 'Y' : 'y', def ? 'n' : 'N'); | 
|  | fflush(stdout); | 
|  |  | 
|  | while (getline(&line, &len, stdin) == -1) { | 
|  | printf("failed to read prompt; assuming '%s'\n", | 
|  | def ? "yes" : "no"); | 
|  | break; | 
|  | } | 
|  |  | 
|  | if (strcmp("\n", line) != 0) { | 
|  | switch (rpmatch(line)) { | 
|  | case 0: ret = false; break; | 
|  | case 1: ret = true; break; | 
|  | case -1: | 
|  | puts("unknown response; please try again"); | 
|  | continue; | 
|  | } | 
|  | } | 
|  | break; | 
|  | } while (1); | 
|  |  | 
|  | free(line); | 
|  |  | 
|  | return ret; | 
|  | } | 
|  |  | 
|  | static inline int is_power_of_2(unsigned long long n) | 
|  | { | 
|  | return (n != 0 && ((n & (n - 1)) == 0)); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * simple_strtoX - convert a hex/dec/oct string into a number | 
|  | * @snum: buffer to convert | 
|  | * @error: set to 1 when buffer isn't fully consumed | 
|  | * | 
|  | * These functions are similar to the standard strtoX() functions, but they are | 
|  | * a little bit easier to use if you want to convert full string of digits into | 
|  | * the binary form. The typical usage: | 
|  | * | 
|  | * int error = 0; | 
|  | * unsigned long num; | 
|  | * | 
|  | * num = simple_strtoul(str, &error); | 
|  | * if (error || ... if needed, your check that num is not out of range ...) | 
|  | * 	error_happened(); | 
|  | */ | 
|  | #define simple_strtoX(func, type) \ | 
|  | static inline type simple_##func(const char *snum, int *error) \ | 
|  | { \ | 
|  | char *endptr; \ | 
|  | type ret = func(snum, &endptr, 0); \ | 
|  | \ | 
|  | if (error && (!*snum || *endptr)) { \ | 
|  | errmsg("%s: unable to parse the number '%s'", #func, snum); \ | 
|  | *error = 1; \ | 
|  | } \ | 
|  | \ | 
|  | return ret; \ | 
|  | } | 
|  | simple_strtoX(strtol, long int) | 
|  | simple_strtoX(strtoll, long long int) | 
|  | simple_strtoX(strtoul, unsigned long int) | 
|  | simple_strtoX(strtoull, unsigned long long int) | 
|  |  | 
|  | /* Simple version-printing for utils */ | 
|  | #define common_print_version() \ | 
|  | do { \ | 
|  | printf("%s %s\n", PROGRAM_NAME, VERSION); \ | 
|  | } while (0) | 
|  |  | 
|  | #include "xalloc.h" | 
|  |  | 
|  | #ifdef __cplusplus | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #endif /* !__MTD_UTILS_COMMON_H__ */ |