2009-03-15 18:02:36 +00:00
|
|
|
/* Copyright (c) 2008-2009, Avian Contributors
|
2008-02-19 18:06:52 +00:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2007-10-23 01:00:57 +00:00
|
|
|
#ifndef X86_H
|
|
|
|
#define X86_H
|
|
|
|
|
|
|
|
#include "types.h"
|
2008-01-01 17:08:47 +00:00
|
|
|
#include "common.h"
|
|
|
|
|
2007-10-23 01:00:57 +00:00
|
|
|
#ifdef __i386__
|
|
|
|
|
2008-06-04 22:21:27 +00:00
|
|
|
# ifdef __APPLE__
|
|
|
|
# if __DARWIN_UNIX03 && defined(_STRUCT_X86_EXCEPTION_STATE32)
|
|
|
|
# define IP_REGISTER(context) (context->uc_mcontext->__ss.__eip)
|
|
|
|
# define BASE_REGISTER(context) (context->uc_mcontext->__ss.__ebp)
|
|
|
|
# define STACK_REGISTER(context) (context->uc_mcontext->__ss.__esp)
|
|
|
|
# define THREAD_REGISTER(context) (context->uc_mcontext->__ss.__ebx)
|
|
|
|
# else
|
|
|
|
# define IP_REGISTER(context) (context->uc_mcontext->ss.eip)
|
|
|
|
# define BASE_REGISTER(context) (context->uc_mcontext->ss.ebp)
|
|
|
|
# define STACK_REGISTER(context) (context->uc_mcontext->ss.esp)
|
|
|
|
# define THREAD_REGISTER(context) (context->uc_mcontext->ss.ebx)
|
|
|
|
# endif
|
|
|
|
# else
|
|
|
|
# define IP_REGISTER(context) (context->uc_mcontext.gregs[REG_EIP])
|
|
|
|
# define BASE_REGISTER(context) (context->uc_mcontext.gregs[REG_EBP])
|
|
|
|
# define STACK_REGISTER(context) (context->uc_mcontext.gregs[REG_ESP])
|
|
|
|
# define THREAD_REGISTER(context) (context->uc_mcontext.gregs[REG_EBX])
|
|
|
|
# endif
|
|
|
|
|
2007-10-23 01:00:57 +00:00
|
|
|
extern "C" uint64_t
|
2007-10-24 17:24:19 +00:00
|
|
|
vmNativeCall(void* function, void* stack, unsigned stackSize,
|
|
|
|
unsigned returnType);
|
2007-10-23 01:00:57 +00:00
|
|
|
|
|
|
|
namespace vm {
|
|
|
|
|
|
|
|
inline uint64_t
|
|
|
|
dynamicCall(void* function, uintptr_t* arguments, uint8_t*,
|
|
|
|
unsigned, unsigned argumentsSize, unsigned returnType)
|
|
|
|
{
|
2007-10-24 17:24:19 +00:00
|
|
|
return vmNativeCall(function, arguments, argumentsSize, returnType);
|
2007-10-23 01:00:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace vm
|
|
|
|
|
|
|
|
#elif defined __x86_64__
|
|
|
|
|
2008-06-04 22:21:27 +00:00
|
|
|
# define IP_REGISTER(context) (context->uc_mcontext.gregs[REG_RIP])
|
|
|
|
# define BASE_REGISTER(context) (context->uc_mcontext.gregs[REG_RBP])
|
|
|
|
# define STACK_REGISTER(context) (context->uc_mcontext.gregs[REG_RSP])
|
|
|
|
# define THREAD_REGISTER(context) (context->uc_mcontext.gregs[REG_RBX])
|
|
|
|
|
2007-10-23 01:00:57 +00:00
|
|
|
extern "C" uint64_t
|
2007-10-24 17:24:19 +00:00
|
|
|
vmNativeCall(void* function, void* stack, unsigned stackSize,
|
|
|
|
void* gprTable, void* sseTable, unsigned returnType);
|
2007-10-23 01:00:57 +00:00
|
|
|
|
|
|
|
namespace vm {
|
|
|
|
|
|
|
|
inline uint64_t
|
|
|
|
dynamicCall(void* function, uint64_t* arguments, uint8_t* argumentTypes,
|
|
|
|
unsigned argumentCount, unsigned, unsigned returnType)
|
|
|
|
{
|
|
|
|
const unsigned GprCount = 6;
|
|
|
|
uint64_t gprTable[GprCount];
|
|
|
|
unsigned gprIndex = 0;
|
|
|
|
|
|
|
|
const unsigned SseCount = 8;
|
|
|
|
uint64_t sseTable[SseCount];
|
|
|
|
unsigned sseIndex = 0;
|
|
|
|
|
|
|
|
uint64_t stack[argumentCount];
|
|
|
|
unsigned stackIndex = 0;
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < argumentCount; ++i) {
|
|
|
|
switch (argumentTypes[i]) {
|
|
|
|
case FLOAT_TYPE:
|
|
|
|
case DOUBLE_TYPE: {
|
|
|
|
if (sseIndex < SseCount) {
|
|
|
|
sseTable[sseIndex++] = arguments[i];
|
|
|
|
} else {
|
|
|
|
stack[stackIndex++] = arguments[i];
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
|
|
|
|
default: {
|
|
|
|
if (gprIndex < GprCount) {
|
|
|
|
gprTable[gprIndex++] = arguments[i];
|
|
|
|
} else {
|
|
|
|
stack[stackIndex++] = arguments[i];
|
|
|
|
}
|
|
|
|
} break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-06-04 22:21:27 +00:00
|
|
|
return vmNativeCall(function, stack, stackIndex * BytesPerWord,
|
2007-10-24 17:24:19 +00:00
|
|
|
(gprIndex ? gprTable : 0),
|
|
|
|
(sseIndex ? sseTable : 0), returnType);
|
2007-10-23 01:00:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace vm
|
|
|
|
|
|
|
|
#else
|
2008-06-04 22:21:27 +00:00
|
|
|
# error unsupported architecture
|
2007-10-23 01:00:57 +00:00
|
|
|
#endif
|
|
|
|
|
2009-03-09 15:29:37 +00:00
|
|
|
namespace vm {
|
|
|
|
|
|
|
|
inline void
|
|
|
|
trap()
|
|
|
|
{
|
|
|
|
asm("int3");
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
memoryBarrier()
|
|
|
|
{
|
|
|
|
__asm__ __volatile__("": : :"memory");
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
storeStoreMemoryBarrier()
|
|
|
|
{
|
|
|
|
memoryBarrier();
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
storeLoadMemoryBarrier()
|
|
|
|
{
|
|
|
|
memoryBarrier();
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
loadMemoryBarrier()
|
|
|
|
{
|
|
|
|
memoryBarrier();
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
syncInstructionCache(const void*, unsigned)
|
|
|
|
{
|
|
|
|
// ignore
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace vm
|
2007-10-23 01:00:57 +00:00
|
|
|
|
|
|
|
#endif//X86_H
|