379 lines
9.8 KiB
C
379 lines
9.8 KiB
C
/////////////////////////////////////////////////////////////////////////
|
|
// $Id$
|
|
/////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Copyright (C) 2001-2018 The Bochs Project
|
|
//
|
|
// This library is free software; you can redistribute it and/or
|
|
// modify it under the terms of the GNU Lesser General Public
|
|
// License as published by the Free Software Foundation; either
|
|
// version 2 of the License, or (at your option) any later version.
|
|
//
|
|
// This library 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
|
|
// Lesser General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Lesser General Public
|
|
// License along with this library; if not, write to the Free Software
|
|
// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
/////////////////////////////////////////////////////////////////////////
|
|
|
|
//
|
|
// osdep.h
|
|
//
|
|
// requires Bit32u/Bit64u from config.h, size_t from stdio.h
|
|
//
|
|
// Operating system dependent includes and defines for Bochs. These
|
|
// declarations can be included by C or C++., but they require definition of
|
|
// size_t beforehand. This makes it difficult to place them into either
|
|
// config.h or bochs.h. If in config.h, size_t is not always available yet.
|
|
// If in bochs.h, they can't be included by C programs so they lose.
|
|
//
|
|
|
|
#ifndef BX_OSDEP_H
|
|
#define BX_OSDEP_H
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif /* __cplusplus */
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// Hacks for win32, but exclude MINGW32 because it doesn't need them.
|
|
//////////////////////////////////////////////////////////////////////
|
|
#ifdef WIN32
|
|
|
|
#ifndef __MINGW32__
|
|
|
|
// Definitions that are needed for WIN32 compilers EXCEPT FOR
|
|
// cygwin compiling with -mno-cygwin. e.g. VC++.
|
|
|
|
#if !defined(_MSC_VER) // gcc without -mno-cygwin
|
|
#define FMT_64 "ll"
|
|
#define FMT_LL "%ll"
|
|
#define FMT_TICK "%011llu"
|
|
#define FMT_ADDRX64 "%016llx"
|
|
#define FMT_PHY_ADDRX64 "%012llx"
|
|
#else
|
|
#define FMT_64 "I64"
|
|
#define FMT_LL "%I64"
|
|
#define FMT_TICK "%011I64u"
|
|
#define FMT_ADDRX64 "%016I64x"
|
|
#define FMT_PHY_ADDRX64 "%012I64x"
|
|
#endif
|
|
|
|
// always return regular file.
|
|
#ifndef S_ISREG
|
|
# define S_ISREG(m) (((m) & S_IFMT) == S_IFREG)
|
|
#endif
|
|
#ifndef S_ISCHR
|
|
# define S_ISCHR(m) (((m) & S_IFMT) == S_IFCHR)
|
|
#endif
|
|
|
|
#if defined(_MSC_VER)
|
|
// win32 has snprintf though with different name.
|
|
#define snprintf _snprintf
|
|
#define vsnprintf _vsnprintf
|
|
#undef BX_HAVE_SNPRINTF
|
|
#undef BX_HAVE_VSNPRINTF
|
|
#define BX_HAVE_SNPRINTF 1
|
|
#define BX_HAVE_VSNPRINTF 1
|
|
|
|
#define access _access
|
|
#define fdopen _fdopen
|
|
#define mktemp _mktemp
|
|
#define off_t __int64
|
|
#define lseek _lseeki64
|
|
#define fseeko64 _fseeki64
|
|
#define fstat _fstati64
|
|
#define stat _stati64
|
|
#define read _read
|
|
#define write _write
|
|
#define open _open
|
|
#define close _close
|
|
#define unlink _unlink
|
|
#define strdup _strdup
|
|
#define strrev _strrev
|
|
#define stricmp _stricmp
|
|
#define getch _getch
|
|
#define strtoll _strtoi64
|
|
#define strtoull _strtoui64
|
|
#define isatty _isatty
|
|
#define fileno _fileno
|
|
#endif
|
|
|
|
#else /* __MINGW32__ defined */
|
|
// Definitions for cygwin compiled with -mno-cygwin
|
|
#define FMT_64 "I64"
|
|
#define FMT_LL "%I64"
|
|
#define FMT_TICK "%011I64u"
|
|
#define FMT_ADDRX64 "%016I64x"
|
|
#define FMT_PHY_ADDRX64 "%012I64x"
|
|
|
|
#define off_t __int64
|
|
// mingw gcc 4.6.1 already has lseek defined
|
|
#ifndef lseek
|
|
#define lseek _lseeki64
|
|
#endif
|
|
#endif /* __MINGW32__ defined */
|
|
|
|
#else /* not WIN32 definitions */
|
|
#if SIZEOF_UNSIGNED_LONG == 8
|
|
#define FMT_64 "l"
|
|
#define FMT_LL "%l"
|
|
#define FMT_TICK "%011lu"
|
|
#define FMT_ADDRX64 "%016lx"
|
|
#define FMT_PHY_ADDRX64 "%012lx"
|
|
#else
|
|
#define FMT_64 "ll"
|
|
#define FMT_LL "%ll"
|
|
#define FMT_TICK "%011llu"
|
|
#define FMT_ADDRX64 "%016llx"
|
|
#define FMT_PHY_ADDRX64 "%012llx"
|
|
#endif
|
|
#endif /* not WIN32 definitions */
|
|
|
|
#define FMT_ADDRX32 "%08x"
|
|
#define FMT_ADDRX16 "%04x"
|
|
|
|
|
|
// Missing defines for open
|
|
#ifndef S_IRUSR
|
|
#define S_IRUSR 0400
|
|
#define S_IWUSR 0200
|
|
#endif
|
|
#ifndef S_IRGRP
|
|
#define S_IRGRP 0040
|
|
#define S_IWGRP 0020
|
|
#endif
|
|
#ifndef S_IROTH
|
|
#define S_IROTH 0004
|
|
#define S_IWOTH 0002
|
|
#endif
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// Missing library functions.
|
|
// These should work on any platform that needs them.
|
|
//
|
|
// A missing library function is renamed to a bx_* function, so that when
|
|
// debugging and linking there's no confusion over which version is used.
|
|
// Because of renaming, the bx_* replacement functions can be tested on
|
|
// machines which have the real library function without duplicate symbols.
|
|
//
|
|
// If you're considering implementing a missing library function, note
|
|
// that it might be cleaner to conditionally disable the function call!
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
#if !BX_HAVE_SNPRINTF
|
|
#define snprintf bx_snprintf
|
|
extern int bx_snprintf (char *s, size_t maxlen, const char *format, ...);
|
|
#endif
|
|
|
|
#if !BX_HAVE_VSNPRINTF
|
|
#define vsnprintf bx_vsnprintf
|
|
extern int bx_vsnprintf (char *s, size_t maxlen, const char *format, va_list arg);
|
|
#endif
|
|
|
|
#if BX_HAVE_STRTOULL
|
|
// great, just use the usual function
|
|
#elif BX_HAVE_STRTOUQ
|
|
// they have strtouq and not strtoull
|
|
#define strtoull strtouq
|
|
#else
|
|
#define strtoull bx_strtoull
|
|
extern Bit64u bx_strtoull (const char *nptr, char **endptr, int baseignore);
|
|
#endif
|
|
|
|
#if !BX_HAVE_STRDUP
|
|
#define strdup bx_strdup
|
|
extern char *bx_strdup(const char *str);
|
|
#endif
|
|
|
|
#if !BX_HAVE_STRREV
|
|
#define strrev bx_strrev
|
|
extern char *bx_strrev(char *str);
|
|
#endif
|
|
|
|
#if BX_HAVE_STRICMP
|
|
// great, just use the usual function
|
|
#elif BX_HAVE_STRCASECMP
|
|
#define stricmp strcasecmp
|
|
#else
|
|
// FIXME: for now using case sensitive function
|
|
#define stricmp strcmp
|
|
#endif
|
|
|
|
#if !BX_HAVE_SOCKLEN_T
|
|
// needed on MacOS X 10.1
|
|
typedef int socklen_t;
|
|
#endif
|
|
|
|
#if !BX_HAVE_SSIZE_T
|
|
// needed on Windows
|
|
typedef Bit64s ssize_t;
|
|
#endif
|
|
|
|
#if !BX_HAVE_MKSTEMP
|
|
#define mkstemp bx_mkstemp
|
|
BOCHSAPI_MSVCONLY extern int bx_mkstemp(char *tpl);
|
|
#endif
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// Missing library functions, implemented for MacOS only
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
#if BX_WITH_MACOS
|
|
// fd_read and fd_write are called by floppy.cc to access the Mac
|
|
// floppy drive directly, since the MacOS doesn't have "special"
|
|
// pathnames which map directly to IO devices
|
|
|
|
int fd_read(char *buffer, Bit32u offset, Bit32u bytes);
|
|
int fd_write(char *buffer, Bit32u offset, Bit32u bytes);
|
|
int fd_stat(struct stat *buf);
|
|
FILE * fdopen(int fd, const char *type);
|
|
|
|
typedef long ssize_t ;
|
|
#endif
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// Missing library functions and byte-swapping stuff,
|
|
// implemented for MorphOS only
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
#ifdef __MORPHOS__
|
|
int fseeko(FILE *stream, off_t offset, int whence);
|
|
struct tm *localtime_r(const time_t *timep, struct tm *result);
|
|
|
|
BX_CPP_INLINE Bit16u bx_ppc_bswap16(Bit16u val)
|
|
{
|
|
Bit32u res;
|
|
|
|
__asm__("rlwimi %0,%0,16,8,15"
|
|
: "=r" (res)
|
|
: "0" (val));
|
|
|
|
return (Bit16u)(res >> 8);
|
|
}
|
|
|
|
BX_CPP_INLINE Bit32u bx_ppc_bswap32(Bit32u val)
|
|
{
|
|
Bit32u res;
|
|
|
|
__asm__("rotlwi %0,%1,8\n\t"
|
|
"rlwimi %0,%1,24,0,7\n\t"
|
|
"rlwimi %0,%1,24,16,23"
|
|
: "=&r" (res)
|
|
: "r" (val));
|
|
|
|
return res;
|
|
}
|
|
|
|
BX_CPP_INLINE Bit64u bx_ppc_bswap64(Bit64u val)
|
|
{
|
|
Bit32u hi, lo;
|
|
|
|
__asm__("rotlwi %0,%2,8\n\t"
|
|
"rlwimi %0,%2,24,0,7\n\t"
|
|
"rlwimi %0,%2,24,16,23\n\t"
|
|
"rotlwi %1,%3,8\n\t"
|
|
"rlwimi %1,%3,24,0,7\n\t"
|
|
"rlwimi %1,%3,24,16,23"
|
|
: "=&r" (hi), "=&r" (lo)
|
|
: "r" ((Bit32u)(val & 0xffffffff)), "r" ((Bit32u)(val >> 32)));
|
|
|
|
return ((Bit64u)hi << 32) | (Bit64u)lo;
|
|
}
|
|
|
|
BX_CPP_INLINE Bit16u bx_ppc_load_le16(const Bit16u *p)
|
|
{
|
|
Bit16u v;
|
|
__asm__("lhbrx %0, 0, %1"
|
|
: "=r" (v)
|
|
: "r" (p), "m" (*p));
|
|
return v;
|
|
}
|
|
|
|
BX_CPP_INLINE void bx_ppc_store_le16(Bit16u *p, Bit16u v)
|
|
{
|
|
__asm__("sthbrx %1, 0, %2"
|
|
: "=m" (*p)
|
|
: "r" (v), "r" (p));
|
|
}
|
|
|
|
BX_CPP_INLINE Bit32u bx_ppc_load_le32(const Bit32u *p)
|
|
{
|
|
Bit32u v;
|
|
__asm__("lwbrx %0, 0, %1"
|
|
: "=r" (v)
|
|
: "r" (p), "m" (*p));
|
|
return v;
|
|
}
|
|
|
|
BX_CPP_INLINE void bx_ppc_store_le32(Bit32u *p, Bit32u v)
|
|
{
|
|
__asm__("stwbrx %1, 0, %2"
|
|
: "=m" (*p)
|
|
: "r" (v), "r" (p));
|
|
}
|
|
|
|
BX_CPP_INLINE Bit64u bx_ppc_load_le64(const Bit64u *p)
|
|
{
|
|
Bit32u hi, lo;
|
|
|
|
__asm__("lwbrx %0, 0, %2\n\t"
|
|
"lwbrx %1, 0, %3"
|
|
: "=&r" (lo), "=&r" (hi)
|
|
: "r" ((Bit32u *)p), "r" ((Bit32u *)p+1));
|
|
|
|
return ((Bit64u)hi << 32) | (Bit64u)lo;
|
|
}
|
|
|
|
BX_CPP_INLINE void bx_ppc_store_le64(Bit64u *p, Bit64u v)
|
|
{
|
|
__asm__("stwbrx %1, 0, %3\n\t"
|
|
"stwbrx %2, 0, %4"
|
|
: "=m" (*p)
|
|
: "r" ((Bit32u)(v & 0xffffffff)), "r" ((Bit32u)(v >> 32)),
|
|
"r" ((Bit32u *)p), "r" ((Bit32u *)p+1));
|
|
}
|
|
#endif
|
|
|
|
//////////////////////////////////////////////////////////////////////
|
|
// New functions to replace library functions
|
|
// with OS-independent versions
|
|
//////////////////////////////////////////////////////////////////////
|
|
|
|
#if BX_HAVE_REALTIME_USEC
|
|
// 64-bit time in useconds.
|
|
BOCHSAPI_MSVCONLY extern Bit64u bx_get_realtime64_usec (void);
|
|
#endif
|
|
|
|
#ifdef WIN32
|
|
#undef BX_HAVE_MSLEEP
|
|
#define BX_HAVE_MSLEEP 1
|
|
#if !defined(__MINGW32__) && !defined(_MSC_VER)
|
|
#define msleep(msec) _sleep(msec)
|
|
#else
|
|
#define msleep(msec) Sleep(msec)
|
|
#endif
|
|
#endif
|
|
|
|
#ifdef __cplusplus
|
|
}
|
|
#endif /* __cplusplus */
|
|
|
|
// these macros required for large ramfile option functionality
|
|
#if BX_HAVE_TMPFILE64 == 0
|
|
#define tmpfile64 tmpfile /* use regular tmpfile() function */
|
|
#endif
|
|
|
|
#if BX_HAVE_FSEEKO64 == 0
|
|
#if BX_HAVE_FSEEK64
|
|
#define fseeko64 fseek64 /* use fseek64() function */
|
|
#elif !defined(_MSC_VER)
|
|
#define fseeko64 fseeko /* use regular fseeko() function */
|
|
#endif
|
|
#endif
|
|
|
|
#endif /* ifdef BX_OSDEP_H */
|