* Copyright 2005-2011, Ingo Weinhold, ingo_weinhold@gmx.de.
* Distributed under the terms of the MIT License.
*/
#include <errno.h>
#include <string.h>
#include <map>
using namespace std;
static map<int, int> sToHaikuErrorMap;
static map<int, int> sToHostErrorMap;
static bool sErrorMapsInitialized = false;
static void
init_error_map()
{
if (sErrorMapsInitialized)
return;
#define ADD_ERROR(error) \
sToHaikuErrorMap[error] = HAIKU_##error; \
sToHostErrorMap[HAIKU_##error] = error;
ADD_ERROR(E2BIG);
ADD_ERROR(ECHILD);
ADD_ERROR(EDEADLK);
ADD_ERROR(EFBIG);
ADD_ERROR(EMLINK);
ADD_ERROR(ENFILE);
ADD_ERROR(ENODEV);
ADD_ERROR(ENOLCK);
ADD_ERROR(ENOSYS);
ADD_ERROR(ENOTTY);
ADD_ERROR(ENXIO);
ADD_ERROR(ESPIPE);
ADD_ERROR(ESRCH);
#ifdef EFPOS
ADD_ERROR(EFPOS);
#endif
#ifdef ESIGPARM
ADD_ERROR(ESIGPARM);
#endif
ADD_ERROR(EDOM);
ADD_ERROR(ERANGE);
ADD_ERROR(EPROTOTYPE);
ADD_ERROR(EPROTONOSUPPORT);
#ifdef EPFNOSUPPORT
ADD_ERROR(EPFNOSUPPORT);
#endif
ADD_ERROR(EAFNOSUPPORT);
ADD_ERROR(EADDRINUSE);
ADD_ERROR(EADDRNOTAVAIL);
ADD_ERROR(ENETDOWN);
ADD_ERROR(ENETUNREACH);
ADD_ERROR(ENETRESET);
ADD_ERROR(ECONNABORTED);
ADD_ERROR(ECONNRESET);
ADD_ERROR(EISCONN);
ADD_ERROR(ENOTCONN);
#ifdef ESHUTDOWN
ADD_ERROR(ESHUTDOWN);
#endif
ADD_ERROR(ECONNREFUSED);
ADD_ERROR(EHOSTUNREACH);
ADD_ERROR(ENOPROTOOPT);
ADD_ERROR(ENOBUFS);
ADD_ERROR(EINPROGRESS);
ADD_ERROR(EALREADY);
ADD_ERROR(EILSEQ);
#ifdef ENOMSG
ADD_ERROR(ENOMSG);
#endif
#ifdef ESTALE
ADD_ERROR(ESTALE);
#endif
ADD_ERROR(EOVERFLOW);
ADD_ERROR(EMSGSIZE);
ADD_ERROR(EOPNOTSUPP);
ADD_ERROR(ENOTSOCK);
#ifdef EHOSTDOWN
ADD_ERROR(EHOSTDOWN);
#endif
#ifdef EBADMSG
ADD_ERROR(EBADMSG);
#endif
#ifdef ECANCELED
ADD_ERROR(ECANCELED);
#endif
ADD_ERROR(EDESTADDRREQ);
#ifdef EDQUOT
ADD_ERROR(EDQUOT);
#endif
#ifdef EIDRM
ADD_ERROR(EIDRM);
#endif
#ifdef EMULTIHOP
ADD_ERROR(EMULTIHOP);
#endif
#ifdef ENODATA
ADD_ERROR(ENODATA);
#endif
#ifdef ENOLINK
ADD_ERROR(ENOLINK);
#endif
#ifdef ENOSR
ADD_ERROR(ENOSR);
#endif
#ifdef ENOSTR
ADD_ERROR(ENOSTR);
#endif
ADD_ERROR(ENOTSUP);
ADD_ERROR(EPROTO);
#ifdef ETIME
ADD_ERROR(ETIME);
#endif
#ifdef ETXTBSY
ADD_ERROR(ETXTBSY);
#endif
ADD_ERROR(ENOMEM);
ADD_ERROR(EACCES);
ADD_ERROR(EINTR);
ADD_ERROR(EIO);
ADD_ERROR(EBUSY);
ADD_ERROR(EFAULT);
ADD_ERROR(ETIMEDOUT);
ADD_ERROR(EAGAIN);
ADD_ERROR(EWOULDBLOCK);
ADD_ERROR(EBADF);
ADD_ERROR(EEXIST);
ADD_ERROR(EINVAL);
ADD_ERROR(ENAMETOOLONG);
ADD_ERROR(ENOENT);
ADD_ERROR(EPERM);
ADD_ERROR(ENOTDIR);
ADD_ERROR(EISDIR);
ADD_ERROR(ENOTEMPTY);
ADD_ERROR(ENOSPC);
ADD_ERROR(EROFS);
ADD_ERROR(EMFILE);
ADD_ERROR(EXDEV);
ADD_ERROR(ELOOP);
ADD_ERROR(ENOEXEC);
ADD_ERROR(EPIPE);
#ifdef ENOATTR
ADD_ERROR(ENOATTR);
#endif
sErrorMapsInitialized = true;
}
static int
to_host_error(int error)
{
init_error_map();
map<int, int>::iterator it = sToHostErrorMap.find(error);
return (it != sToHostErrorMap.end() ? it->second : error);
}
static int
to_haiku_error(int error)
{
init_error_map();
map<int, int>::iterator it = sToHaikuErrorMap.find(error);
if (it != sToHaikuErrorMap.end())
return it->second;
return (error > 0 ? -error : error);
}
char *
_haiku_build_strerror(int errnum)
{
return strerror(to_host_error(errnum));
}
int *
_haiku_build_errno()
{
static int previousErrno = 0;
static int localErrno = 0;
static int previousLocalErrno = 0;
int currentErrno = errno;
if (currentErrno == previousErrno) {
if (localErrno != previousLocalErrno) {
errno = previousErrno = to_host_error(localErrno);
previousLocalErrno = localErrno;
}
} else {
previousErrno = currentErrno;
previousLocalErrno = localErrno = to_haiku_error(errno);
}
return &localErrno;
}
int
_haiku_to_host_error(int error)
{
return to_host_error(error);
}