diff dyncallback/dyncall_callback_x86.c @ 0:3e629dc19168

initial from svn dyncall-1745
author Daniel Adler
date Thu, 19 Mar 2015 22:24:28 +0100
parents
children d48a8b8d2ef9
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/dyncallback/dyncall_callback_x86.c	Thu Mar 19 22:24:28 2015 +0100
@@ -0,0 +1,282 @@
+/*
+
+ Package: dyncall
+ Library: dyncallback
+ File: dyncallback/dyncall_callback_x86.c
+ Description: Callback - Implementation for x86
+ License:
+
+   Copyright (c) 2007-2015 Daniel Adler <dadler@uni-goettingen.de>,
+                           Tassilo Philipp <tphilipp@potion-studios.com>
+
+   Permission to use, copy, modify, and 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.
+
+   THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
+   WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
+   MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
+   ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
+   WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
+   ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
+   OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
+
+*/
+
+
+
+#include "dyncall_callback_x86.h"
+#include "dyncall_args_x86.h"
+
+#include "dyncall_alloc_wx.h"
+#include "dyncall_signature.h"
+
+/*
+ * assembly thunk entry for callbacks
+ */
+
+extern void dcCallbackThunkEntry();
+
+/* compute stacksize for callee cleanup calling conventions:
+ *
+ * stdcall,fastcall_ms,fastcall_gnu
+ */
+
+static int dcbCleanupSize_x86_cdecl(const char* signature)
+{
+  return 0;
+}
+
+static int dcbCleanupSize_x86_std(const char* signature)
+{
+  const char* ptr = signature;
+  int size = 0;
+  char ch;
+  while( (ch = *ptr++) != DC_SIGCHAR_ENDARG ) {
+    switch(ch) {
+      case DC_SIGCHAR_BOOL:
+      case DC_SIGCHAR_CHAR:
+      case DC_SIGCHAR_SHORT:
+      case DC_SIGCHAR_INT:
+      case DC_SIGCHAR_LONG:
+      case DC_SIGCHAR_POINTER:
+      case DC_SIGCHAR_UCHAR:
+      case DC_SIGCHAR_USHORT:
+      case DC_SIGCHAR_UINT:
+      case DC_SIGCHAR_ULONG:
+      case DC_SIGCHAR_STRING:
+      case DC_SIGCHAR_FLOAT:
+        size += 4;
+        break;
+      case DC_SIGCHAR_DOUBLE:
+      case DC_SIGCHAR_LONGLONG:
+      case DC_SIGCHAR_ULONGLONG:
+        size += 8;
+        break;
+    }
+  }
+  return size;
+}
+
+static int dcbCleanupSize_x86_this_ms(const char* signature)
+{
+  const char* ptr = signature;
+  int size = 0;
+  char ch;
+  while( (ch = *ptr++) != DC_SIGCHAR_ENDARG )
+  {
+    switch(ch)
+    {
+    case DC_SIGCHAR_BOOL:
+    case DC_SIGCHAR_CHAR:
+    case DC_SIGCHAR_SHORT:
+    case DC_SIGCHAR_INT:
+    case DC_SIGCHAR_LONG:
+    case DC_SIGCHAR_POINTER:
+    case DC_SIGCHAR_UCHAR:
+    case DC_SIGCHAR_USHORT:
+    case DC_SIGCHAR_UINT:
+    case DC_SIGCHAR_ULONG:
+    case DC_SIGCHAR_STRING:
+    case DC_SIGCHAR_FLOAT:
+      size += 4;
+      break;
+    case DC_SIGCHAR_DOUBLE:
+    case DC_SIGCHAR_LONGLONG:
+    case DC_SIGCHAR_ULONGLONG:
+      size += 8;
+      break;
+    }
+  }
+  return size;
+}
+
+static int dcbCleanupSize_x86_fast_ms(const char* signature)
+{
+  const char* ptr = signature;
+  int size = 0;
+  int regs = 0;
+  char ch;
+  while( (ch = *ptr++) != DC_SIGCHAR_ENDARG )
+  {
+    switch(ch)
+    {
+    case DC_SIGCHAR_BOOL:
+    case DC_SIGCHAR_CHAR:
+    case DC_SIGCHAR_SHORT:
+    case DC_SIGCHAR_INT:
+    case DC_SIGCHAR_LONG:
+    case DC_SIGCHAR_POINTER:
+    case DC_SIGCHAR_UCHAR:
+    case DC_SIGCHAR_USHORT:
+    case DC_SIGCHAR_UINT:
+    case DC_SIGCHAR_ULONG:
+    case DC_SIGCHAR_STRING:
+      if (regs < 2) regs++;
+      else size += 4;
+      break;
+    case DC_SIGCHAR_FLOAT:
+      size += 4;
+      break;
+    case DC_SIGCHAR_DOUBLE:
+      size += 8;
+      break;
+    case DC_SIGCHAR_LONGLONG:
+    case DC_SIGCHAR_ULONGLONG:
+      size += 8;
+      break;
+    }
+  }
+  return size;
+}
+
+static int dcbCleanupSize_x86_fast_gnu(const char* signature)
+{
+  const char* ptr = signature;
+  char ch;
+  int size = 0;
+  int regs = 0;
+  while( (ch = *ptr++) != DC_SIGCHAR_ENDARG ) {
+    switch(ch) {
+      case DC_SIGCHAR_FLOAT:
+        size += 4;
+        break;
+      case DC_SIGCHAR_DOUBLE:
+        size += 8;
+        break;
+      case DC_SIGCHAR_LONGLONG:
+      case DC_SIGCHAR_ULONGLONG:
+        regs  = 2;
+        size += 8;
+        break;
+      default:
+        if (regs < 2) regs++;
+        else size += 4;
+        break;
+    }
+  }
+  return size;
+}
+
+void dcbInitCallback(DCCallback* pcb, const char* signature, DCCallbackHandler* handler, void* userdata)
+{
+  const char* ptr;
+  char  ch;
+  int mode;
+  pcb->handler = handler;
+  pcb->userdata = userdata;
+
+  ptr = signature;
+  ch = *ptr;
+
+  /* x86 hints: */
+
+  mode = DC_CALL_C_X86_CDECL;
+
+  if(ch == DC_SIGCHAR_CC_PREFIX)
+  {
+    ptr++;
+    ch = *ptr++;
+    switch(ch) {
+      case DC_SIGCHAR_CC_STDCALL:      mode = DC_CALL_C_X86_WIN32_STD;      break;
+      case DC_SIGCHAR_CC_THISCALL_MS:  mode = DC_CALL_C_X86_WIN32_THIS_MS;  break;
+      case DC_SIGCHAR_CC_FASTCALL_GNU: mode = DC_CALL_C_X86_WIN32_FAST_GNU; break;
+      case DC_SIGCHAR_CC_FASTCALL_MS:  mode = DC_CALL_C_X86_WIN32_FAST_MS;  break;
+    }
+  }
+
+  /* x86 configuration: */
+
+  switch(mode) {
+    case DC_CALL_C_X86_CDECL:
+      pcb->args_vt = &dcArgsVT_default;
+      pcb->stack_cleanup = dcbCleanupSize_x86_cdecl(ptr);
+      break;
+    case DC_CALL_C_X86_WIN32_STD:
+      pcb->args_vt = &dcArgsVT_default;
+      pcb->stack_cleanup = dcbCleanupSize_x86_std(ptr);
+      break;
+    case DC_CALL_C_X86_WIN32_THIS_MS:
+      pcb->args_vt = &dcArgsVT_this_ms;
+      pcb->stack_cleanup = dcbCleanupSize_x86_this_ms(ptr);
+      break;
+    case DC_CALL_C_X86_WIN32_FAST_MS:
+      pcb->args_vt = &dcArgsVT_fast_ms;
+      pcb->stack_cleanup = dcbCleanupSize_x86_fast_ms(ptr);
+      break;
+    case DC_CALL_C_X86_WIN32_FAST_GNU:
+      pcb->args_vt = &dcArgsVT_fast_gnu;
+      pcb->stack_cleanup = dcbCleanupSize_x86_fast_gnu(ptr);
+      break;
+  }
+
+#if defined(DC_PLAN9)
+  /* HACK for Plan9 - 'reuse' pcb->stack_cleanup as a flag
+     to indicate if return value is 64bit. The field is not
+     used anyways as the caller is responsible to clean up
+     the stack in Plan9. If set to '1' the callback kernel
+     takes into account an extra stack-parameter (pointer
+     to 64bit return value).
+     I thought of introducing a new field, but for one single
+     x86 calling convention out of many, it seemed overkill
+     to change the struct for everybody else. Maybe renaming
+     would be a good idea, though. ~ Tassilo
+  */
+  while(*ptr) {
+    if(*ptr++ == DC_SIGCHAR_ENDARG) {
+      pcb->stack_cleanup = (*ptr == DC_SIGCHAR_LONGLONG) || (*ptr == DC_SIGCHAR_ULONGLONG);
+      break;
+    }
+  }
+#endif
+}
+
+/*
+ * callback constructor
+ */
+
+DCCallback* dcbNewCallback(const char* signature, DCCallbackHandler* handler, void* userdata)
+{
+  int err;
+  DCCallback* pcb;
+  err = dcAllocWX(sizeof(DCCallback), (void**) &pcb);
+  if (err != 0) return 0;
+
+  dcbInitThunk(&pcb->thunk, dcCallbackThunkEntry);
+  dcbInitCallback(pcb, signature, handler, userdata);
+  return pcb;
+}
+
+/*
+ * free
+ */
+
+void dcbFreeCallback(DCCallback* pcb)
+{
+  dcFreeWX(pcb, sizeof(DCCallback));
+}
+
+void* dcbGetUserData(DCCallback* pcb)
+{
+  return pcb->userdata;
+}