Logo AND Algorithmique Numérique Distribuée

Public GIT Repository
751289047d528ce09503b2ed10829f896b5c4d03
[simgrid.git] / src / bindings / lua / lua_task.c
1 /* Copyright (c) 2010. The SimGrid Team.
2  * All rights reserved.                                                     */
3
4 /* This program is free software; you can redistribute it and/or modify it
5  * under the terms of the license (GNU LGPL) which comes with this package. */
6
7 #include "lua_private.h"
8 #include "lua_utils.h"
9 #include "lua_state_cloner.h"
10 #include <lauxlib.h>
11
12 XBT_LOG_NEW_DEFAULT_SUBCATEGORY(lua_task, bindings, "Lua bindings (task module)");
13
14 #define TASK_MODULE_NAME "simgrid.task"
15
16 /* ********************************************************************************* */
17 /*                                simgrid.task API                                   */
18 /* ********************************************************************************* */
19
20 /**
21  * \brief Ensures that a value in the stack is a valid task and returns it.
22  * \param L a Lua state
23  * \param index an index in the Lua stack
24  * \return the C task corresponding to this Lua task
25  */
26 m_task_t sglua_check_task(lua_State* L, int index)
27 {
28   sglua_stack_dump("check task: ", L);
29   luaL_checktype(L, index, LUA_TTABLE);
30                                   /* ... task ... */
31   lua_getfield(L, index, "__simgrid_task");
32                                   /* ... task ... ctask */
33   m_task_t task = *((m_task_t*) luaL_checkudata(L, -1, TASK_MODULE_NAME));
34   lua_pop(L, 1);
35                                   /* ... task ... */
36
37   if (task == NULL) {
38     luaL_error(L, "This task was sent to someone else, you cannot access it anymore");
39   }
40
41   return task;
42 }
43
44 /**
45  * \brief Creates a new task and leaves it onto the stack.
46  * \param L a Lua state
47  * \return number of values returned to Lua
48  *
49  * - Argument 1 (string): name of the task
50  * - Argument 2 (number): computation size
51  * - Argument 3 (number): communication size
52  * - Return value (task): the task created
53  *
54  * A Lua task is a regular table with a full userdata inside, and both share
55  * the same metatable. For the regular table, the metatable allows OO-style
56  * writing such as your_task:send(someone).
57  * For the userdata, the metatable is used to check its type.
58  * TODO: make the task name an optional last parameter
59  */
60 static int l_task_new(lua_State* L)
61 {
62   XBT_DEBUG("Task new");
63   const char* name = luaL_checkstring(L, 1);
64   int comp_size = luaL_checkint(L, 2);
65   int msg_size = luaL_checkint(L, 3);
66                                   /* name comp comm */
67   lua_settop(L, 0);
68                                   /* -- */
69   m_task_t msg_task = MSG_task_create(name, comp_size, msg_size, NULL);
70
71   lua_newtable(L);
72                                   /* task */
73   luaL_getmetatable(L, TASK_MODULE_NAME);
74                                   /* task mt */
75   lua_setmetatable(L, -2);
76                                   /* task */
77   m_task_t* lua_task = (m_task_t*) lua_newuserdata(L, sizeof(m_task_t));
78                                   /* task ctask */
79   *lua_task = msg_task;
80   luaL_getmetatable(L, TASK_MODULE_NAME);
81                                   /* task ctask mt */
82   lua_setmetatable(L, -2);
83                                   /* task ctask */
84   lua_setfield(L, -2, "__simgrid_task");
85                                   /* task */
86   return 1;
87 }
88
89 /**
90  * \brief Returns the name of a task.
91  * \param L a Lua state
92  * \return number of values returned to Lua
93  *
94  * - Argument 1 (task): a task
95  * - Return value (string): name of the task
96  */
97 static int l_task_get_name(lua_State* L)
98 {
99   m_task_t task = sglua_check_task(L, 1);
100   lua_pushstring(L, MSG_task_get_name(task));
101   return 1;
102 }
103
104 /**
105  * \brief Returns the computation duration of a task.
106  * \param L a Lua state
107  * \return number of values returned to Lua
108  *
109  * - Argument 1 (task): a task
110  * - Return value (number): computation duration of this task
111  */
112 static int l_task_get_computation_duration(lua_State* L)
113 {
114   m_task_t task = sglua_check_task(L, 1);
115   lua_pushnumber(L, MSG_task_get_compute_duration(task));
116   return 1;
117 }
118
119 /**
120  * \brief Executes a task.
121  * \param L a Lua state
122  * \return number of values returned to Lua
123  *
124  * - Argument 1 (task): the task to execute
125  * - Return value (nil or string): nil if the task was successfully executed,
126  * or an error string in case of failure, which may be "task canceled" or
127  * "host failure"
128  */
129 static int l_task_execute(lua_State* L)
130 {
131   m_task_t task = sglua_check_task(L, 1);
132   MSG_error_t res = MSG_task_execute(task);
133
134   if (res == MSG_OK) {
135     return 0;
136   }
137   else {
138     lua_pushstring(L, sglua_get_msg_error(res));
139     return 1;
140   }
141 }
142
143 /**
144  * \brief Pops the Lua task from the stack and registers it so that the
145  * process can retrieve it later knowing the C task.
146  * \param L a lua state
147  */
148 void sglua_task_register(lua_State* L) {
149
150   m_task_t task = sglua_check_task(L, -1);
151                                   /* ... task */
152   /* put in the C task a ref to the lua task so that the receiver finds it */
153   unsigned long ref = luaL_ref(L, LUA_REGISTRYINDEX);
154                                   /* ... */
155   MSG_task_set_data(task, (void*) ref);
156 }
157
158 /**
159  * \brief Pushes onto the stack the Lua task corresponding to a C task.
160  *
161  * The Lua task must have been previously registered with task_register so
162  * that it can be retrieved knowing the C task.
163  *
164  * \param L a lua state
165  * \param task a C task
166  */
167 void sglua_task_unregister(lua_State* L, m_task_t task) {
168
169                                   /* ... */
170   /* the task is in my registry, put it onto my stack */
171   unsigned long ref = (unsigned long) MSG_task_get_data(task);
172   lua_rawgeti(L, LUA_REGISTRYINDEX, ref);
173                                   /* ... task */
174   luaL_unref(L, LUA_REGISTRYINDEX, ref);
175   MSG_task_set_data(task, NULL);
176 }
177
178 /**
179  * \brief This function is called when a C task has just been copied.
180  *
181  * This callback is used to move the corresponding Lua task from the sender
182  * process to the receiver process.
183  * It is executed in SIMIX kernel mode when the communication finishes,
184  * before both processes are awaken. Thus, this function is thread-safe when
185  * user processes are executed in parallel, though it modifies the Lua
186  * stack of both processes to move the task.
187  * After this function, both Lua stacks are restored in their previous state.
188  * The task is moved from the registry of the sender to the registry of the
189  * receiver.
190  *
191  * \param task the task copied
192  * \param src_process the sender
193  * \param dst_process the receiver
194  */
195 static void task_copy_callback(m_task_t task, m_process_t src_process,
196     m_process_t dst_process) {
197
198   lua_State* src = MSG_process_get_data(src_process);
199   lua_State* dst = MSG_process_get_data(dst_process);
200
201                                   /* src: ...
202                                      dst: ... */
203   sglua_task_unregister(src, task);
204                                   /* src: ... task */
205   sglua_copy_value(src, dst);
206                                   /* src: ... task
207                                      dst: ... task */
208   sglua_task_register(dst);
209                                   /* dst: ... */
210
211   /* the receiver is now the owner of the task and may destroy it:
212    * make the sender forget the C task so that it doesn't garbage */
213   lua_getfield(src, -1, "__simgrid_task");
214                                   /* src: ... task ctask */
215   m_task_t* udata = (m_task_t*) luaL_checkudata(src, -1, TASK_MODULE_NAME);
216   *udata = NULL;
217   lua_pop(src, 2);
218                                   /* src: ... */
219 }
220
221 /**
222  * \brief Sends a task to a mailbox and waits for its completion.
223  * \param L a Lua state
224  * \return number of values returned to Lua
225  *
226  * - Argument 1 (task): the task to send
227  * - Argument 2 (string or compatible): mailbox name, as a real string or any
228  * type convertible to string (numbers always are)
229  * - Return values (boolean + string): true if the communication was successful,
230  * or false plus an error string in case of failure, which may be "timeout",
231  * "host failure" or "transfer failure"
232  */
233 static int l_task_send(lua_State* L)
234 {
235   m_task_t task = sglua_check_task(L, 1);
236   const char* mailbox = luaL_checkstring(L, 2);
237                                   /* task mailbox ... */
238   lua_settop(L, 1);
239                                   /* task */
240   sglua_task_register(L);
241                                   /* -- */
242   MSG_error_t res = MSG_task_send(task, mailbox);
243
244   if (res == MSG_OK) {
245     lua_pushboolean(L, 1);
246                                   /* true */
247     return 1;
248   }
249   else {
250     /* the communication has failed, I'm still the owner of the task */
251     sglua_task_unregister(L, task);
252                                   /* task */
253     lua_pushboolean(L, 0);
254                                   /* task false */
255     lua_pushstring(L, sglua_get_msg_error(res));
256                                   /* task false error */
257     return 2;
258   }
259 }
260
261 /**
262  * \brief Sends a task on a mailbox.
263  * \param L a Lua state
264  * \return number of values returned to Lua
265  *
266  * This is a non-blocking function: use simgrid.comm.wait() or
267  * simgrid.comm.test() to end the communication.
268  *
269  * - Argument 1 (task): the task to send
270  * - Argument 2 (string or compatible): mailbox name, as a real string or any
271  * type convertible to string (numbers always are)
272  * - Return value (comm): a communication object to be used later with wait or test
273  */
274 static int l_task_isend(lua_State* L)
275 {
276   m_task_t task = sglua_check_task(L, 1);
277   const char* mailbox = luaL_checkstring(L, 2);
278                                   /* task mailbox ... */
279   lua_settop(L, 1);
280                                   /* task */
281   sglua_task_register(L);
282                                   /* -- */
283   msg_comm_t comm = MSG_task_isend(task, mailbox);
284
285   sglua_push_comm(L, comm);
286                                   /* comm */
287   return 1;
288 }
289
290 /**
291  * \brief Sends a task on a mailbox on a best effort way (detached send).
292  * \param L a Lua state
293  * \return number of values returned to Lua
294  *
295  * Like simgrid.task.isend, this is a non-blocking function.
296  * You can use this function if you don't care about when the communication
297  * ends and whether it succeeds.
298  * FIXME: isn't this equivalent to calling simgrid.task.isend() and ignoring
299  * the result?
300  *
301  * - Argument 1 (task): the task to send
302  * - Argument 2 (string or compatible): mailbox name, as a real string or any
303  * type convertible to string (numbers always are)
304  */
305 static int l_task_dsend(lua_State* L)
306 {
307   m_task_t task = sglua_check_task(L, 1);
308   const char* mailbox = luaL_checkstring(L, 2);
309                                   /* task mailbox ... */
310   lua_settop(L, 1);
311                                   /* task */
312   sglua_task_register(L);
313                                   /* -- */
314   MSG_task_dsend(task, mailbox, NULL);
315   return 0;
316 }
317
318 /**
319  * \brief Receives a task.
320  * \param L a Lua state
321  * \return number of values returned to Lua
322  *
323  * - Argument 1 (string or compatible): mailbox name, as a real string or any
324  * type convertible to string (numbers always are)
325  * - Argument 2 (number, optional): timeout (default is no timeout)
326  * - Return values (task or nil + string): the task received, or nil plus an
327  * error message if the communication has failed
328  */
329 static int l_task_recv(lua_State* L)
330 {
331   m_task_t task = NULL;
332   const char* mailbox = luaL_checkstring(L, 1);
333   int timeout;
334   if (lua_gettop(L) >= 2) {
335                                   /* mailbox timeout ... */
336     timeout = luaL_checknumber(L, 2);
337   }
338   else {
339                                   /* mailbox */
340     timeout = -1;
341     /* no timeout by default */
342   }
343                                   /* mailbox ... */
344   MSG_error_t res = MSG_task_receive_with_timeout(&task, mailbox, timeout);
345
346   if (res == MSG_OK) {
347     sglua_task_unregister(L, task);
348                                   /* mailbox ... task */
349     return 1;
350   }
351   else {
352     lua_pushnil(L);
353                                   /* mailbox ... nil */
354     lua_pushstring(L, sglua_get_msg_error(res));
355                                   /* mailbox ... nil error */
356     return 2;
357   }
358 }
359
360 /**
361  * \brief Asynchronously receives a task on a mailbox.
362  * \param L a Lua state
363  * \return number of values returned to Lua
364  *
365  * This is a non-blocking function: use simgrid.comm.wait() or
366  * simgrid.comm.test() to end the communication and get the task in case of
367  * success.
368  *
369  * - Argument 1 (string or compatible): mailbox name, as a real string or any
370  * type convertible to string (numbers always are)
371  * - Return value (comm): a communication object to be used later with wait or test
372  */
373
374 static int l_task_irecv(lua_State* L)
375 {
376   const char* mailbox = luaL_checkstring(L, 1);
377                                   /* mailbox ... */
378   m_task_t* task = xbt_new0(m_task_t, 1); // FIXME fix this leak
379   msg_comm_t comm = MSG_task_irecv(task, mailbox);
380   sglua_push_comm(L, comm);
381                                   /* mailbox ... comm */
382   return 1;
383 }
384
385 static const luaL_reg task_functions[] = {
386   {"new", l_task_new},
387   {"get_name", l_task_get_name},
388   {"get_computation_duration", l_task_get_computation_duration},
389   {"execute", l_task_execute},
390   {"send", l_task_send},
391   {"isend", l_task_isend},
392   {"dsend", l_task_dsend},
393   {"recv", l_task_recv},
394   {"irecv", l_task_irecv},
395   {NULL, NULL}
396 };
397
398 /**
399  * \brief Finalizes the userdata of a task.
400  * \param L a Lua state
401  * \return number of values returned to Lua
402  *
403  * - Argument 1 (userdata): a C task, possibly NULL if it was sent to another
404  * Lua state
405  */
406 static int l_task_gc(lua_State* L)
407 {
408                                   /* ctask */
409   m_task_t task = *((m_task_t*) luaL_checkudata(L, 1, TASK_MODULE_NAME));
410   /* the task is NULL if I sent it to someone else */
411   if (task != NULL) {
412     MSG_task_destroy(task);
413   }
414   return 0;
415 }
416
417 /**
418  * \brief Returns a string representation of a C task.
419  * \param L a Lua state
420  * \return number of values returned to Lua
421  *
422  * - Argument 1 (userdata): a task
423  * - Return value (string): a string describing this task
424  */
425 static int l_task_tostring(lua_State* L)
426 {
427   m_task_t task = *((m_task_t*) luaL_checkudata(L, 1, TASK_MODULE_NAME));
428   lua_pushfstring(L, "Task: %p", task);
429   return 1;
430 }
431
432 /**
433  * \brief Metamethods of both a task table and the userdata inside it.
434  */
435 static const luaL_reg task_meta[] = {
436   {"__gc", l_task_gc}, /* will be called only for userdata */
437   {"__tostring", l_task_tostring},
438   {NULL, NULL}
439 };
440
441 /**
442  * \brief Registers the task functions into the table simgrid.task.
443  *
444  * Also initialize the metatable of the task userdata type.
445  *
446  * \param L a lua state
447  */
448 void sglua_register_task_functions(lua_State* L)
449 {
450   /* create a table simgrid.task and fill it with task functions */
451   luaL_openlib(L, TASK_MODULE_NAME, task_functions, 0);
452                                   /* simgrid.task */
453
454   /* create the metatable for tasks, add it to the Lua registry */
455   luaL_newmetatable(L, TASK_MODULE_NAME);
456                                   /* simgrid.task mt */
457   /* fill the metatable */
458   luaL_openlib(L, NULL, task_meta, 0);
459                                   /* simgrid.task mt */
460   lua_pushvalue(L, -2);
461                                   /* simgrid.task mt simgrid.task */
462   /* metatable.__index = simgrid.task
463    * we put the task functions inside the task itself:
464    * this allows to write my_task:method(args) for
465    * simgrid.task.method(my_task, args) */
466   lua_setfield(L, -2, "__index");
467                                   /* simgrid.task mt */
468   lua_pop(L, 2);
469                                   /* -- */
470
471   /* set up MSG to copy Lua tasks between states */
472   MSG_task_set_copy_callback(task_copy_callback);
473 }
474