Logo AND Algorithmique Numérique Distribuée

Public GIT Repository
For win64 it is cast unsigned long long.
[simgrid.git] / src / xbt / win32_ucontext.c
1 /*\r
2  *      win32-ucontext: Unix ucontext_t operations on Windows platforms\r
3  *      Copyright(C) 2007 Panagiotis E. Hadjidoukas\r
4  *\r
5  *      Contact Email: phadjido@cs.uoi.gr, xdoukas@ceid.upatras.gr\r
6  *\r
7  *      win32-ucontext is free software; you can redistribute it and/or\r
8  *      modify it under the terms of the GNU Lesser General Public\r
9  *      License as published by the Free Software Foundation; either\r
10  *      version 2 of the License, or (at your option) any later version.\r
11  *\r
12  *      win32-ucontext is distributed in the hope that it will be useful,\r
13  *      but WITHOUT ANY WARRANTY; without even the implied warranty of\r
14  *      MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\r
15  *      Lesser General Public License for more details.\r
16  *\r
17  *      You should have received a copy of the GNU Lesser General Public\r
18  *      License along with SimGrid in the file LICENSE-LGPL-2.1;\r
19  *      if not, write to the Free Software Foundation, Inc.,\r
20  *      59 Temple Place - Suite 330, Boston, MA 02111-1307, USA\r
21  */  \r
22     \r
23 #include "win32_ucontext.h"\r
24 int getcontext(ucontext_t * ucp) \r
25 {\r
26   int ret;\r
27   \r
28       /* Retrieve the full machine context */ \r
29       ucp->uc_mcontext.ContextFlags = CONTEXT_FULL;\r
30   ret = GetThreadContext(GetCurrentThread(), &ucp->uc_mcontext);\r
31   return (ret == 0) ? -1 : 0;\r
32 }\r
33 \r
34 int setcontext(const ucontext_t * ucp) \r
35 {\r
36   int ret;\r
37   \r
38       /* Restore the full machine context (already set) */ \r
39       ret = SetThreadContext(GetCurrentThread(), &ucp->uc_mcontext);\r
40   return (ret == 0) ? -1 : 0;\r
41 }\r
42 \r
43 int makecontext(ucontext_t * ucp, void (*func) (), int argc, ...) \r
44 {\r
45   int i;\r
46   va_list ap;\r
47   char *sp;\r
48   \r
49       /* Stack grows down */ \r
50       sp = (char *) (size_t) ucp->uc_stack.ss_sp + ucp->uc_stack.ss_size;\r
51   \r
52       /* Reserve stack space for the arguments (maximum possible: argc*(8 bytes per argument)) */ \r
53       sp -= argc * 8;\r
54   if (sp < (char *) ucp->uc_stack.ss_sp) {\r
55     \r
56         /* errno = ENOMEM; */ \r
57         return -1;\r
58   }\r
59   \r
60       /* Set the instruction and the stack pointer */\r
61   #ifdef I_X86_\r
62   ucp->uc_mcontext.Eip = (unsigned long) func;\r
63   ucp->uc_mcontext.Esp = (unsigned long) sp - 4;\r
64   #endif\r
65   #ifdef _IA64_\r
66   #  error "_IA64_"\r
67   #endif\r
68   #ifdef _AMD64_\r
69   ucp->uc_mcontext.Rip = (unsigned long long) func;\r
70   ucp->uc_mcontext.Rsp = (unsigned long long) sp - 8;\r
71   #endif\r
72 \r
73       /* Save/Restore the full machine context */ \r
74       ucp->uc_mcontext.ContextFlags = CONTEXT_FULL;\r
75   \r
76       /* Copy the arguments */ \r
77       va_start(ap, argc);\r
78   for (i = 0; i < argc; i++) {\r
79     memcpy(sp, ap, 8);\r
80     ap += 8;\r
81     sp += 8;\r
82   }\r
83   va_end(ap);\r
84   return 0;\r
85 }\r
86 \r
87 int swapcontext(ucontext_t * oucp, const ucontext_t * ucp) \r
88 {\r
89   int ret;\r
90   if ((oucp == NULL) || (ucp == NULL)) {\r
91     \r
92         /*errno = EINVAL; */ \r
93         return -1;\r
94   }\r
95   ret = getcontext(oucp);\r
96   if (ret == 0) {\r
97     ret = setcontext(ucp);\r
98   }\r
99   return ret;\r
100 }\r
101 \r