2005-03-18 23:07:55 +01:00
|
|
|
// Copyright (c) 2003-2005 Jan Wassenberg
|
2004-03-03 00:56:51 +01:00
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
//
|
|
|
|
// Contact info:
|
|
|
|
// Jan.Wassenberg@stud.uni-karlsruhe.de
|
|
|
|
// http://www.stud.uni-karlsruhe.de/~urkt/
|
|
|
|
|
2005-03-18 23:07:55 +01:00
|
|
|
/*
|
|
|
|
|
|
|
|
[KEEP IN SYNC WITH WIKI]
|
|
|
|
|
|
|
|
low-level aka "lib"
|
|
|
|
-------------------
|
|
|
|
|
|
|
|
this codebase was grown from modules shared between several projects,
|
|
|
|
i.e. my personal library; hence the name "lib". it has been expanded to
|
|
|
|
fit the needs of 0ad - in particular, resource loading.
|
|
|
|
|
|
|
|
owing to the dual-use situation, the 0ad coding conventions are not met;
|
|
|
|
also, major changes are ill-advised because they may break other projects.
|
|
|
|
|
|
|
|
|
|
|
|
design goals
|
|
|
|
------------
|
|
|
|
|
|
|
|
- fast and low-overhead, including startup time
|
|
|
|
- portable: must run on Win32, Mac OS X and Linux
|
|
|
|
- reusable across projects, i.e. no dependency on a
|
|
|
|
central 'manager' that ties modules together.
|
|
|
|
|
|
|
|
|
|
|
|
scope
|
|
|
|
-----
|
|
|
|
|
|
|
|
- POSIX definitions
|
|
|
|
- resource management
|
|
|
|
- debugging tools (including memory tracker)
|
|
|
|
- low-level helper functions, e.g. ADTs, endian conversion and timing
|
|
|
|
- platform-dependent system/feature detection
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef LIB_H__
|
|
|
|
#define LIB_H__
|
2004-03-03 00:56:51 +01:00
|
|
|
|
|
|
|
#include <stddef.h>
|
2004-06-28 17:34:10 +02:00
|
|
|
#include <assert.h>
|
2004-03-03 00:56:51 +01:00
|
|
|
|
|
|
|
#include "config.h"
|
2005-02-26 16:15:36 +01:00
|
|
|
#include "lib/types.h"
|
2004-03-03 00:56:51 +01:00
|
|
|
|
|
|
|
#include "sysdep/sysdep.h"
|
|
|
|
|
|
|
|
|
2004-05-29 14:04:42 +02:00
|
|
|
// tell STL not to generate exceptions, if compiling without exceptions
|
|
|
|
// (usually done for performance reasons).
|
|
|
|
#ifdef CONFIG_DISABLE_EXCEPTIONS
|
|
|
|
# ifdef _WIN32
|
|
|
|
# define _HAS_EXCEPTIONS 0
|
|
|
|
# else
|
|
|
|
# define STL_NO_EXCEPTIONS
|
|
|
|
# endif
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
2004-06-01 19:34:12 +02:00
|
|
|
|
2004-05-06 19:14:30 +02:00
|
|
|
#define STMT(STMT_code__) do { STMT_code__; } while(0)
|
2004-03-03 00:56:51 +01:00
|
|
|
|
2005-04-10 00:24:08 +02:00
|
|
|
// may be called at any time (in particular before main), but is not
|
|
|
|
// thread-safe. if that's important, use pthread_once() instead.
|
2004-03-03 00:56:51 +01:00
|
|
|
#define ONCE(ONCE_code__)\
|
2004-05-06 19:14:30 +02:00
|
|
|
STMT(\
|
2004-03-03 00:56:51 +01:00
|
|
|
static bool ONCE_done__ = false;\
|
2004-06-01 19:34:12 +02:00
|
|
|
if(!ONCE_done__)\
|
2004-03-03 00:56:51 +01:00
|
|
|
{\
|
|
|
|
ONCE_done__ = true;\
|
|
|
|
ONCE_code__;\
|
|
|
|
}\
|
2004-05-06 19:14:30 +02:00
|
|
|
)
|
|
|
|
|
2005-04-10 00:24:08 +02:00
|
|
|
|
2005-01-07 01:55:53 +01:00
|
|
|
// note: UINT_MAX is necessary when testing a Handle value and
|
|
|
|
// also returning Handle. the negative value (error return)
|
|
|
|
// is guaranteed to fit into an int, but we need to "mask"
|
|
|
|
// it to avoid VC cast-to-smaller-type warnings.
|
2004-05-06 19:14:30 +02:00
|
|
|
|
2004-06-02 17:12:48 +02:00
|
|
|
#ifdef _WIN32
|
2004-05-06 19:14:30 +02:00
|
|
|
#define CHECK_ERR(func)\
|
|
|
|
STMT(\
|
2005-01-07 01:55:53 +01:00
|
|
|
int err__ = (int)((func) & UINT_MAX);\
|
2004-06-21 16:16:10 +02:00
|
|
|
if(err__ < 0)\
|
2004-05-13 15:52:48 +02:00
|
|
|
{\
|
2005-03-27 03:31:39 +02:00
|
|
|
assert2(0 && "FYI: CHECK_ERR reports that a function failed."\
|
2004-05-13 15:52:48 +02:00
|
|
|
"feel free to ignore or suppress this warning.");\
|
2004-06-21 16:16:10 +02:00
|
|
|
return err__;\
|
2004-05-13 15:52:48 +02:00
|
|
|
}\
|
2004-05-06 19:14:30 +02:00
|
|
|
)
|
2004-06-02 17:12:48 +02:00
|
|
|
#else
|
|
|
|
#define CHECK_ERR(func)\
|
|
|
|
STMT(\
|
2005-01-07 01:55:53 +01:00
|
|
|
int err__ = (int)((func) & UINT_MAX);\
|
2004-06-21 16:16:10 +02:00
|
|
|
if(err__ < 0)\
|
2004-06-02 17:12:48 +02:00
|
|
|
{\
|
2005-05-12 00:40:19 +02:00
|
|
|
debug_printf("%s:%d: FYI: CHECK_ERR reports that a function failed."\
|
2004-06-09 15:46:06 +02:00
|
|
|
"feel free to ignore or suppress this warning.\n", __FILE__, __LINE__);\
|
2004-06-21 16:16:10 +02:00
|
|
|
return err__;\
|
2004-06-02 17:12:48 +02:00
|
|
|
}\
|
|
|
|
)
|
|
|
|
#endif
|
2004-03-03 00:56:51 +01:00
|
|
|
|
2005-03-27 19:40:40 +02:00
|
|
|
// just pass on errors without any kind of annoying warning
|
|
|
|
// (useful for functions that can legitimately fail, e.g. vfs_exists).
|
|
|
|
#define RETURN_ERR(func)\
|
|
|
|
STMT(\
|
|
|
|
int err__ = (int)((func) & UINT_MAX);\
|
|
|
|
if(err__ < 0)\
|
|
|
|
return err__;\
|
|
|
|
)
|
|
|
|
|
2005-03-22 03:17:55 +01:00
|
|
|
#define THROW_ERR(func)\
|
|
|
|
STMT(\
|
|
|
|
int err__ = (int)((func) & UINT_MAX);\
|
|
|
|
if(err__ < 0)\
|
|
|
|
{\
|
|
|
|
assert(0 && "FYI: CHECK_ERR reports that a function failed."\
|
|
|
|
"feel free to ignore or suppress this warning.");\
|
|
|
|
throw err__;\
|
|
|
|
}\
|
|
|
|
)
|
|
|
|
|
2004-06-08 14:10:51 +02:00
|
|
|
|
|
|
|
|
2005-03-18 23:07:55 +01:00
|
|
|
// useful because VC6 may return 0 on failure, instead of throwing.
|
2004-06-18 15:24:59 +02:00
|
|
|
// this wraps the exception handling, and creates a NULL pointer on failure.
|
2004-06-08 14:10:51 +02:00
|
|
|
#define SAFE_NEW(type, ptr)\
|
|
|
|
type* ptr;\
|
|
|
|
try\
|
|
|
|
{\
|
|
|
|
ptr = new type();\
|
|
|
|
}\
|
|
|
|
catch(std::bad_alloc)\
|
|
|
|
{\
|
|
|
|
ptr = 0;\
|
2004-06-18 15:24:59 +02:00
|
|
|
}
|
2004-06-08 14:10:51 +02:00
|
|
|
|
|
|
|
|
2004-07-14 05:06:15 +02:00
|
|
|
|
2004-03-03 00:56:51 +01:00
|
|
|
enum LibError
|
|
|
|
{
|
2004-12-01 22:37:01 +01:00
|
|
|
//
|
|
|
|
// lib + res
|
|
|
|
//
|
|
|
|
|
|
|
|
ERR_INVALID_PARAM = -1000,
|
|
|
|
ERR_INVALID_HANDLE = -1001,
|
|
|
|
ERR_NO_MEM = -1002,
|
|
|
|
|
|
|
|
// try again later
|
|
|
|
ERR_AGAIN = -1003,
|
|
|
|
|
|
|
|
// fixed limit exceeded
|
|
|
|
ERR_LIMIT = -1004,
|
|
|
|
|
|
|
|
// system doesn't support required API(s)
|
|
|
|
ERR_NO_SYS = -1005,
|
|
|
|
|
|
|
|
// feature not currently implemented (will probably change)
|
|
|
|
ERR_NOT_IMPLEMENTED = -1006,
|
|
|
|
|
|
|
|
// feature won't be supported
|
|
|
|
ERR_NOT_SUPPORTED = -1007,
|
|
|
|
|
|
|
|
// file contents are damaged
|
|
|
|
ERR_CORRUPTED = -1008,
|
|
|
|
|
|
|
|
ERR_UNKNOWN_FORMAT = -1009,
|
|
|
|
|
2005-01-10 14:42:10 +01:00
|
|
|
ERR_TIMED_OUT = -1010,
|
|
|
|
|
2004-12-01 22:37:01 +01:00
|
|
|
//
|
|
|
|
// file + vfs
|
|
|
|
//
|
|
|
|
|
|
|
|
ERR_FILE_NOT_FOUND = -1100,
|
|
|
|
ERR_PATH_NOT_FOUND = -1101,
|
|
|
|
ERR_DIR_END = -1102,
|
|
|
|
ERR_EOF = -1103,
|
|
|
|
ERR_PATH_LENGTH = -1104,
|
|
|
|
ERR_NOT_FILE = -1105,
|
|
|
|
ERR_FILE_ACCESS = -1106,
|
|
|
|
ERR_IO = -1107,
|
|
|
|
|
|
|
|
|
|
|
|
ERR_TEX_FMT_INVALID = -1200,
|
|
|
|
|
2004-03-03 00:56:51 +01:00
|
|
|
|
2004-05-06 19:14:30 +02:00
|
|
|
ERR_LAST
|
2004-03-03 00:56:51 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2004-11-23 22:08:59 +01:00
|
|
|
|
|
|
|
|
2004-03-03 00:56:51 +01:00
|
|
|
#ifndef MIN
|
|
|
|
#define MIN(a, b) (((a) < (b))? (a) : (b))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#ifndef MAX
|
|
|
|
#define MAX(a, b) (((a) > (b))? (a) : (b))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
2005-03-18 23:07:55 +01:00
|
|
|
// 2 ways of avoiding "unreferenced formal parameter" warnings:
|
|
|
|
// .. inside the function body, e.g. void f(int x) { UNUSED(x); }
|
2004-03-03 00:56:51 +01:00
|
|
|
#define UNUSED(param) (void)param;
|
2005-03-18 23:07:55 +01:00
|
|
|
// .. wrapped around the parameter name, e.g. void f(int UNUSEDPARAM(x))
|
2004-07-24 16:04:40 +02:00
|
|
|
#define UNUSEDPARAM(param)
|
2004-03-03 00:56:51 +01:00
|
|
|
|
|
|
|
|
2004-10-06 16:11:55 +02:00
|
|
|
#define ARRAY_SIZE(name) (sizeof(name) / sizeof(name[0]))
|
|
|
|
|
2004-03-03 00:56:51 +01:00
|
|
|
|
|
|
|
//
|
|
|
|
// compile-time assert, especially useful for testing sizeof().
|
|
|
|
// no runtime overhead; may be used anywhere, including file scope.
|
|
|
|
//
|
|
|
|
|
|
|
|
// generate a symbol containing the line number of the macro invocation.
|
|
|
|
// used to give a unique name (per file) to types made by cassert.
|
2004-05-06 19:14:30 +02:00
|
|
|
// we can't prepend __FILE__ to make it globally unique - the filename
|
|
|
|
// may be enclosed in quotes. need the 2 macro expansions to make sure
|
|
|
|
// __LINE__ is expanded correctly.
|
2004-03-03 00:56:51 +01:00
|
|
|
#define MAKE_UID2__(l) LINE_ ## l
|
|
|
|
#define MAKE_UID1__(l) MAKE_UID2__(l)
|
|
|
|
#define UID__ MAKE_UID1__(__LINE__)
|
|
|
|
|
|
|
|
// more descriptive error message, but may cause a struct redefinition
|
|
|
|
// warning if used from the same line in different files.
|
2004-06-22 15:22:46 +02:00
|
|
|
#define cassert(expr) struct UID__ { int CASSERT_FAILURE: (expr); }
|
2004-03-03 00:56:51 +01:00
|
|
|
|
|
|
|
// less helpful error message, but redefinition doesn't trigger warnings.
|
2004-06-22 15:22:46 +02:00
|
|
|
#define cassert2(expr) extern char CASSERT_FAILURE[1][(expr)]
|
2004-03-03 00:56:51 +01:00
|
|
|
|
|
|
|
// note: alternative method in C++: specialize a struct only for true;
|
2004-05-06 19:14:30 +02:00
|
|
|
// using it will raise 'incomplete type' errors if instantiated with false.
|
|
|
|
|
2004-03-03 00:56:51 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2005-01-23 18:54:20 +01:00
|
|
|
const size_t KiB = 1ul << 10;
|
|
|
|
const size_t MiB = 1ul << 20;
|
|
|
|
const size_t GiB = 1ul << 30;
|
2004-03-03 00:56:51 +01:00
|
|
|
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
#define DIR_SEP '\\'
|
|
|
|
#else
|
|
|
|
#define DIR_SEP '/'
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
#define BIT(n) (1ul << (n))
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2004-06-01 19:34:12 +02:00
|
|
|
|
|
|
|
// FNV1-A hash - good for strings.
|
|
|
|
// if len = 0 (default), treat buf as a C-string;
|
|
|
|
// otherwise, hash <len> bytes of buf.
|
|
|
|
extern u32 fnv_hash(const void* buf, const size_t len = 0);
|
2005-03-18 23:07:55 +01:00
|
|
|
extern u64 fnv_hash64(const void* buf, const size_t len = 0);
|
2004-06-01 19:34:12 +02:00
|
|
|
|
2005-03-27 03:31:39 +02:00
|
|
|
// special version for strings: first converts to lowercase
|
|
|
|
// (useful for comparing mixed-case filenames)
|
|
|
|
extern u32 fnv_lc_hash(const char* str, const size_t len = 0);
|
|
|
|
|
2004-06-01 19:34:12 +02:00
|
|
|
// hash (currently FNV) of a filename
|
|
|
|
typedef u32 FnHash;
|
|
|
|
|
|
|
|
|
|
|
|
extern u16 addusw(u16 x, u16 y);
|
|
|
|
extern u16 subusw(u16 x, u16 y);
|
|
|
|
|
2005-05-27 06:40:29 +02:00
|
|
|
// zero-extend <size> (truncated to 8) bytes of little-endian data to u64,
|
|
|
|
// starting at address <p> (need not be aligned).
|
|
|
|
extern u64 movzx_64le(const u8* p, size_t size);
|
|
|
|
|
|
|
|
// sign-extend <size> (truncated to 8) bytes of little-endian data to i64,
|
|
|
|
// starting at address <p> (need not be aligned).
|
|
|
|
extern i64 movsx_64le(const u8* p, size_t size);
|
2004-06-01 19:34:12 +02:00
|
|
|
|
|
|
|
|
|
|
|
extern bool is_pow2(long n);
|
|
|
|
|
|
|
|
// return -1 if not an integral power of 2,
|
|
|
|
// otherwise the base2 logarithm
|
|
|
|
extern int ilog2(const int n);
|
|
|
|
|
2004-06-19 16:43:31 +02:00
|
|
|
// return log base 2, rounded up.
|
|
|
|
extern uint log2(uint x);
|
|
|
|
|
2004-06-01 19:34:12 +02:00
|
|
|
|
2005-05-03 07:05:16 +02:00
|
|
|
// multiple must be a power of two.
|
2004-06-01 19:34:12 +02:00
|
|
|
extern uintptr_t round_up(uintptr_t val, uintptr_t multiple);
|
|
|
|
|
|
|
|
extern u16 fp_to_u16(double in);
|
|
|
|
|
2005-04-26 18:47:48 +02:00
|
|
|
// return random integer in [0, limit).
|
|
|
|
// does not use poorly distributed lower bits of rand().
|
|
|
|
extern int rand_up_to(int limit);
|
|
|
|
|
2004-06-01 19:34:12 +02:00
|
|
|
// big endian!
|
|
|
|
extern void base32(const int len, const u8* in, u8* out);
|
|
|
|
|
|
|
|
|
2004-11-23 21:52:03 +01:00
|
|
|
// case-insensitive check if string <s> matches the pattern <w>,
|
|
|
|
// which may contain '?' or '*' wildcards. if so, return 1, otherwise 0.
|
2005-03-27 03:31:39 +02:00
|
|
|
// note: NULL wildcard pattern matches everything!
|
2004-11-23 21:52:03 +01:00
|
|
|
extern int match_wildcard(const char* s, const char* w);
|
|
|
|
|
2004-03-03 00:56:51 +01:00
|
|
|
|
2005-03-18 23:07:55 +01:00
|
|
|
#endif // #ifndef LIB_H__
|