diff options
author | Benoit Germain <benoit.germain@ubisoft.com> | 2024-05-07 17:56:10 +0200 |
---|---|---|
committer | Benoit Germain <benoit.germain@ubisoft.com> | 2024-05-07 17:56:10 +0200 |
commit | 6d271c5796eae14d1dc60e778435495ebfb540d8 (patch) | |
tree | 3bccba196595305ffd5f2b30f838dd39fbc5d51d /docs | |
parent | 5c7ef34404d3367542275d76b49f276ab035639f (diff) | |
download | lanes-6d271c5796eae14d1dc60e778435495ebfb540d8.tar.gz lanes-6d271c5796eae14d1dc60e778435495ebfb540d8.tar.bz2 lanes-6d271c5796eae14d1dc60e778435495ebfb540d8.zip |
Linda API changes
* timeout clarifications (negative values are no longer accepted, use nil instead)
* linda(send, linda.null, key, ...) removed, if you want to send a nil, just do it as usual
Diffstat (limited to 'docs')
-rw-r--r-- | docs/index.html | 51 |
1 files changed, 30 insertions, 21 deletions
diff --git a/docs/index.html b/docs/index.html index e811074..3c9cbcf 100644 --- a/docs/index.html +++ b/docs/index.html | |||
@@ -953,26 +953,29 @@ | |||
953 | </pre></td></tr></table> | 953 | </pre></td></tr></table> |
954 | 954 | ||
955 | <p> | 955 | <p> |
956 | Waits until the lane finishes, or <tt>timeout</tt> seconds have passed. Returns <tt>nil, "timeout"</tt> on timeout, <tt>nil,err,stack_tbl</tt> if the lane hit an error, <tt>nil, "killed"</tt> if forcefully killed, or the return values of the lane. | 956 | <tt>timeout</tt> is an optional number >= 0 (the default if unspecified). |
957 | <br/> | ||
958 | Waits until the lane finishes, or <tt>timeout</tt> seconds have passed. | ||
959 | <br/> | ||
960 | Returns <tt>nil, "timeout"</tt> on timeout, <tt>nil,err,stack_tbl</tt> if the lane hit an error, <tt>nil, "killed"</tt> if forcefully killed, or the return values of the lane. | ||
961 | <br/> | ||
957 | Unlike in reading the results in table fashion, errors are not propagated. | 962 | Unlike in reading the results in table fashion, errors are not propagated. |
958 | </p> | 963 | </p> |
959 | 964 | ||
960 | <p> | 965 | <p> |
961 | <tt>stack_tbl</tt> is a table describing where the error was thrown. | 966 | <tt>stack_tbl</tt> is a table describing where the error was thrown. |
962 | <br/> | 967 | <br/> |
963 | In <tt>"extended"</tt> mode, <tt>stack_tbl</tt> is an array of tables containing info gathered with <tt>lua_getinfo()</tt> (<tt>"source"</tt>,<tt>"currentline"</tt>,<tt>"name"</tt>,<tt>"namewhat"</tt>,<tt>"what"</tt>). | 968 | In <tt>"extended"</tt> mode, <tt>stack_tbl</tt> is an array of tables containing info gathered with <tt>lua_getinfo()</tt> (<tt>"source"</tt>,<tt>"currentline"</tt>,<tt>"name"</tt>,<tt>"namewhat"</tt>,<tt>"what"</tt>). |
964 | <br/> | 969 | <br/> |
965 | In <tt>"basic mode"</tt>, <tt>stack_tbl</tt> is an array of "<filename>:<line>" strings. Use <tt>table.concat()</tt> to format it to your liking (or just ignore it). | 970 | In <tt>"basic"</tt> mode, <tt>stack_tbl</tt> is an array of <tt>"<filename>:<line>"</tt> strings. Use <tt>table.concat()</tt> to format it to your liking (or just ignore it). |
966 | </p> | 971 | </p> |
967 | 972 | ||
968 | <p> | 973 | <p> |
969 | If you use <tt>:join</tt>, make sure your lane main function returns a non-nil value so you can tell timeout and error cases apart from succesful return (using the <tt>.status</tt> property may be risky, since it might change between a timed out join and the moment you read it). | 974 | If you use <tt>:join()</tt>, make sure your lane main function returns a non-nil value so you can tell timeout and error cases apart from succesful return (using the <tt>.status</tt> property may be risky, since it might change between a timed out join and the moment you read it). |
970 | </p> | 975 | </p> |
971 | 976 | ||
972 | <p> | ||
973 | |||
974 | <table border=1 bgcolor="#FFFFE0" cellpadding="10" style="width:50%"><tr><td><pre> | 977 | <table border=1 bgcolor="#FFFFE0" cellpadding="10" style="width:50%"><tr><td><pre> |
975 | require "lanes".configure() | 978 | local lanes = require "lanes".configure() |
976 | 979 | ||
977 | f = lanes.gen(function() error "!!!" end) | 980 | f = lanes.gen(function() error "!!!" end) |
978 | a = f(1) | 981 | a = f(1) |
@@ -990,7 +993,7 @@ | |||
990 | </p> | 993 | </p> |
991 | 994 | ||
992 | <table border=1 bgcolor="#FFFFE0" cellpadding="10" style="width:50%"><tr><td><pre> | 995 | <table border=1 bgcolor="#FFFFE0" cellpadding="10" style="width:50%"><tr><td><pre> |
993 | require "lanes".configure() | 996 | local lanes = require "lanes".configure() |
994 | 997 | ||
995 | local sync_linda = lanes.linda() | 998 | local sync_linda = lanes.linda() |
996 | f = lanes.gen(function() dostuff() sync_linda:send("done", true) end) | 999 | f = lanes.gen(function() dostuff() sync_linda:send("done", true) end) |
@@ -1012,7 +1015,10 @@ | |||
1012 | </pre></td></tr></table> | 1015 | </pre></td></tr></table> |
1013 | 1016 | ||
1014 | <p> | 1017 | <p> |
1015 | <tt>cancel()</tt> sends a cancellation request to the lane.<br/> | 1018 | <tt>timeout</tt> is an optional number >= 0. Defaults to 0 if left unspecified or <tt>nil</tt>. |
1019 | <br/> | ||
1020 | <tt>cancel()</tt> sends a cancellation request to the lane. | ||
1021 | <br/> | ||
1016 | First argument is a <tt>mode</tt> can be one of <tt>"hard"</tt>, <tt>"soft"</tt>, <tt>"call"</tt>, <tt>"ret"</tt>, <tt>"line"</tt>, <tt>"count"</tt>. | 1022 | First argument is a <tt>mode</tt> can be one of <tt>"hard"</tt>, <tt>"soft"</tt>, <tt>"call"</tt>, <tt>"ret"</tt>, <tt>"line"</tt>, <tt>"count"</tt>. |
1017 | If <tt>mode</tt> is not specified, it defaults to <tt>"hard"</tt>. | 1023 | If <tt>mode</tt> is not specified, it defaults to <tt>"hard"</tt>. |
1018 | If <tt>wake_lane</tt> is <tt>true</tt>, the lane is also signalled so that execution returns from any pending linda operation. Linda operations detecting the cancellation request return <tt>lanes.cancel_error</tt>. | 1024 | If <tt>wake_lane</tt> is <tt>true</tt>, the lane is also signalled so that execution returns from any pending linda operation. Linda operations detecting the cancellation request return <tt>lanes.cancel_error</tt>. |
@@ -1056,17 +1062,17 @@ | |||
1056 | </pre></td></tr></table> | 1062 | </pre></td></tr></table> |
1057 | 1063 | ||
1058 | <p> | 1064 | <p> |
1059 | The <tt>error</tt> call is used for throwing exceptions in Lua. What Lua does not offer, however, is scoped <a href="http://en.wikipedia.org/wiki/Finalizer">finalizers</a> | 1065 | The regular Lua <tt>error</tt> function is usable in lanes for throwing exceptions. What Lua does not offer, however, is scoped <a href="http://en.wikipedia.org/wiki/Finalizer">finalizers</a> |
1060 | that would get called when a certain block of instructions gets exited, whether through peaceful return or abrupt <tt>error</tt>. | 1066 | that would get called when a certain block of instructions gets exited, whether through peaceful return or abrupt <tt>error</tt>. |
1061 | </p> | 1067 | </p> |
1062 | 1068 | ||
1063 | <p> | 1069 | <p> |
1064 | Since 2.0.3, Lanes registers a function <tt>set_finalizer</tt> in the lane's Lua state for doing this. | 1070 | Lanes registers a function <tt>set_finalizer</tt> in the lane's Lua state for doing this. |
1065 | Any functions given to it will be called in the lane Lua state, just prior to closing it. It is possible to set more than one finalizer. They are not called in any particular order. | 1071 | Any functions given to it will be called in the lane Lua state, just prior to closing it. It is possible to set more than one finalizer. They are not called in any particular order. |
1066 | </p> | 1072 | </p> |
1067 | 1073 | ||
1068 | <p> | 1074 | <p> |
1069 | An error in a finalizer itself overrides the state of the regular chunk (in practise, it would be highly preferable <i>not</i> to have errors in finalizers). If one finalizer errors, the others may not get called. | 1075 | An error in a finalizer itself overrides the state of the regular chunk (in practice, it would be highly preferable <i>not</i> to have errors in finalizers). If one finalizer errors, the others may not get called. |
1070 | If a finalizer error occurs after an error in the lane body, then this new error replaces the previous one (including the full stack trace). | 1076 | If a finalizer error occurs after an error in the lane body, then this new error replaces the previous one (including the full stack trace). |
1071 | </p> | 1077 | </p> |
1072 | 1078 | ||
@@ -1103,18 +1109,18 @@ | |||
1103 | </p> | 1109 | </p> |
1104 | 1110 | ||
1105 | <table border="1" bgcolor="#FFFFE0" cellpadding="10" style="width:50%"><tr><td><pre> | 1111 | <table border="1" bgcolor="#FFFFE0" cellpadding="10" style="width:50%"><tr><td><pre> |
1106 | require "lanes".configure() | 1112 | local lanes = require "lanes".configure() |
1107 | 1113 | ||
1108 | local linda = lanes.linda() | 1114 | local linda = lanes.linda("my linda") |
1109 | 1115 | ||
1110 | local function loop(max) | 1116 | local function loop(max) |
1111 | for i = 1, max do | 1117 | for i = 1, max do |
1112 | print("sending: " .. i) | 1118 | print("sending: " .. i) |
1113 | linda:send("x", i) -- linda as upvalue | 1119 | linda:send("x", i) -- linda as upvalue of loop() |
1114 | end | 1120 | end |
1115 | end | 1121 | end |
1116 | 1122 | ||
1117 | a = lanes.gen("", loop)(10000) | 1123 | lane_h = lanes.gen("", loop)(10000) |
1118 | 1124 | ||
1119 | while true do | 1125 | while true do |
1120 | local key, val = linda:receive(3.0, "x") -- timeout in seconds | 1126 | local key, val = linda:receive(3.0, "x") -- timeout in seconds |
@@ -1124,6 +1130,8 @@ | |||
1124 | end | 1130 | end |
1125 | print(tostring(linda) .. " received: " .. val) | 1131 | print(tostring(linda) .. " received: " .. val) |
1126 | end | 1132 | end |
1133 | |||
1134 | lane_h:join() | ||
1127 | </pre></td></tr></table> | 1135 | </pre></td></tr></table> |
1128 | 1136 | ||
1129 | <p> | 1137 | <p> |
@@ -1147,7 +1155,7 @@ | |||
1147 | <table border="1" bgcolor="#E0E0FF" cellpadding="10" style="width:50%"><tr><td><pre> | 1155 | <table border="1" bgcolor="#E0E0FF" cellpadding="10" style="width:50%"><tr><td><pre> |
1148 | h = lanes.linda([opt_name, [opt_group]]) | 1156 | h = lanes.linda([opt_name, [opt_group]]) |
1149 | 1157 | ||
1150 | [true|lanes.cancel_error] = h:send([timeout_secs,] [h.null,] key, ...) | 1158 | [true|lanes.cancel_error] = h:send([timeout_secs,] key, ...) |
1151 | 1159 | ||
1152 | [key, val]|[lanes.cancel_error] = h:receive([timeout_secs,] key [, ...]) | 1160 | [key, val]|[lanes.cancel_error] = h:receive([timeout_secs,] key [, ...]) |
1153 | 1161 | ||
@@ -1157,10 +1165,11 @@ | |||
1157 | </pre></td></tr></table> | 1165 | </pre></td></tr></table> |
1158 | 1166 | ||
1159 | <p> | 1167 | <p> |
1160 | The <tt>send()</tt> and <tt>receive()</tt> methods use Linda keys as FIFO stacks (first in, first out). Timeouts are given in seconds (millisecond accuracy). If using numbers as the first Linda key, one must explicitly give <tt>nil</tt> as the timeout parameter to avoid ambiguities. | 1168 | Timeouts are given in seconds (>= 0, millisecond accuracy) or <tt>nil</tt>. Timeout can be omitted only if the first key is not a number (then it's equivalent to an infinite duration). |
1161 | </p> | 1169 | </p> |
1162 | 1170 | ||
1163 | <p> | 1171 | <p> |
1172 | The <tt>send()</tt> and <tt>receive()</tt> methods use Linda keys as FIFO stacks (first in, first out).<br/> | ||
1164 | By default, stack sizes are unlimited but limits can be enforced using the <tt>limit()</tt> method. This can be useful to balance execution speeds in a producer/consumer scenario. Any negative value removes the limit. | 1173 | By default, stack sizes are unlimited but limits can be enforced using the <tt>limit()</tt> method. This can be useful to balance execution speeds in a producer/consumer scenario. Any negative value removes the limit. |
1165 | <br/> | 1174 | <br/> |
1166 | A limit of 0 is allowed to block everything. | 1175 | A limit of 0 is allowed to block everything. |
@@ -1181,7 +1190,7 @@ | |||
1181 | <br/> | 1190 | <br/> |
1182 | <tt>send()</tt> returns <tt>lanes.cancel_error</tt> if interrupted by a soft cancel request. | 1191 | <tt>send()</tt> returns <tt>lanes.cancel_error</tt> if interrupted by a soft cancel request. |
1183 | <br/> | 1192 | <br/> |
1184 | If no data is provided after the key, <tt>send()</tt> raises an error. If provided with <tt>linda.null</tt> or <tt>lanes.null</tt> before the actual key and there is no data to send, <tt>send()</tt> sends a single <tt>nil</tt>. | 1193 | If no data is provided after the key, <tt>send()</tt> raises an error. |
1185 | <br/> | 1194 | <br/> |
1186 | Also, if <tt>linda.null</tt> or <tt>lanes.null</tt> is sent as data in a linda, it will be read as a <tt>nil</tt>. | 1195 | Also, if <tt>linda.null</tt> or <tt>lanes.null</tt> is sent as data in a linda, it will be read as a <tt>nil</tt>. |
1187 | </p> | 1196 | </p> |
@@ -1395,7 +1404,7 @@ events to a common Linda, but... :).</font> | |||
1395 | </p> | 1404 | </p> |
1396 | 1405 | ||
1397 | <table border="1" bgcolor="#E0E0FF" cellpadding="10" style="width:50%"><tr><td><pre> | 1406 | <table border="1" bgcolor="#E0E0FF" cellpadding="10" style="width:50%"><tr><td><pre> |
1398 | void = lanes.sleep(['indefinitely'|seconds|false]) | 1407 | void = lanes.sleep(['indefinitely'|seconds|nil]) |
1399 | </pre></td></tr></table> | 1408 | </pre></td></tr></table> |
1400 | 1409 | ||
1401 | <p> | 1410 | <p> |
@@ -1775,7 +1784,7 @@ static MyDeepFactory g_MyDeepFactory; | |||
1775 | <ul> | 1784 | <ul> |
1776 | <li>Data passing (parameters, upvalues, Linda messages) is generally fast, doing two binary state-to-state copies (from source state to hidden state, hidden state to target state). Remember that not only the function you specify but also its upvalues, their upvalues, etc. etc. will get copied.</li> | 1785 | <li>Data passing (parameters, upvalues, Linda messages) is generally fast, doing two binary state-to-state copies (from source state to hidden state, hidden state to target state). Remember that not only the function you specify but also its upvalues, their upvalues, etc. etc. will get copied.</li> |
1777 | <li>Lane startup is fast (1000's of lanes a second), depending on the number of standard libraries initialized. Initializing all standard libraries is about 3-4 times slower than having no standard libraries at all. If you throw in a lot of lanes per second, make sure you give them minimal necessary set of libraries.</li> | 1786 | <li>Lane startup is fast (1000's of lanes a second), depending on the number of standard libraries initialized. Initializing all standard libraries is about 3-4 times slower than having no standard libraries at all. If you throw in a lot of lanes per second, make sure you give them minimal necessary set of libraries.</li> |
1778 | <li>Waiting Lindas are woken up (and execute some hidden Lua code) each time <u>any</u> key in the Lindas they are waiting for are changed. This may give essential slow-down (not measured, just a gut feeling) if a lot of Linda keys are used. Using separate Linda objects for logically separate issues will help (which is good practise anyhow).</li> | 1787 | <li>Waiting Lindas are woken up (and execute some hidden Lua code) each time <u>any</u> key in the Lindas they are waiting for are changed. This may give essential slow-down (not measured, just a gut feeling) if a lot of Linda keys are used. Using separate Linda objects for logically separate issues will help (which is good practice anyhow).</li> |
1779 | <li>Linda objects are light. The memory footprint is two OS-level signalling objects (<tt>HANDLE</tt> or <tt>pthread_cond_t</tt>) for each, plus one C pointer for the proxies per each Lua state using the Linda. Barely nothing.</li> | 1788 | <li>Linda objects are light. The memory footprint is two OS-level signalling objects (<tt>HANDLE</tt> or <tt>pthread_cond_t</tt>) for each, plus one C pointer for the proxies per each Lua state using the Linda. Barely nothing.</li> |
1780 | <li>Timers are light. You can probably expect timers up to 0.01 second resolution to be useful, but that is very system specific. All timers are merged into one main timer state (see <tt>timer.lua</tt>); no OS side timers are utilized.</li> | 1789 | <li>Timers are light. You can probably expect timers up to 0.01 second resolution to be useful, but that is very system specific. All timers are merged into one main timer state (see <tt>timer.lua</tt>); no OS side timers are utilized.</li> |
1781 | <li>If you are using a lot of Linda objects, it may be useful to try having more of these keeper states. By default, only one is used (see <a href="#initialization"><tt>lanes.configure()</tt></a>).</li> | 1790 | <li>If you are using a lot of Linda objects, it may be useful to try having more of these keeper states. By default, only one is used (see <a href="#initialization"><tt>lanes.configure()</tt></a>).</li> |