From cf9b09ff352611fb8cee3679127f3655cbe73ae7 Mon Sep 17 00:00:00 2001
From: Benoit Germain
- This document was revised on 16-Jan-14, and applies to version 3.7.8. + This document was revised on 20-Jan-14, and applies to version 3.8.0.
@@ -1068,23 +1068,31 @@- [true] = linda_h:set( key, [val]) + [bool] = linda_h:set( key [, val [, ...]]) - [val] = linda_h:get( key) + [val [, ...]] = linda_h:get( key [, count = 1]) |
The table access methods are for accessing a slot without queuing or consuming. They can be used for making shared tables of storage among the lanes.
+
+ Writing to a slot never blocks because it ignores the limit. It overwrites existing value and clears any possible queued entries.
+
+ Reading doesn't block either because get() returns whatever is available (which can be nothing), up to the specified count.
+
+ Table access and send()/receive() can be used together; reading a slot essentially peeks the next outcoming value of a queue.
- Writing to a slot never blocks because it ignores the limit. It overwrites existing value and clears any possible queued entries.
+ set() signals the linda for write if a value is stored.
- set() signals the linda from write if a value is stored.
-
- (Since version 3.7.7) if the key was full but the new data count of the key after set() is below its limit, set() returns true and the linda is also signaled for read so that send()-blocked threads are awakened.
+ (Since version 3.7.7) if the key was full but the new data count of the key after set() is below its limit, set() returns true and the linda is also signaled for read so that send()-blocked threads are awakened.
+
+ Since version 3.8.0, set() can write several values at the specified key, writing nil values is now possible, and clearing the contents at the specified key is done by not providing any value.
- Table access and send()/receive() can be used together; reading a slot essentially peeks the next outcoming value of a queue.
+ Also, get() can read several values at once. If the key contains no data, get() returns no value. This can be used to separate the case when reading stored nil values.
diff --git a/src/keeper.c b/src/keeper.c index 6d5ecbf..b0db8b4 100644 --- a/src/keeper.c +++ b/src/keeper.c @@ -115,11 +115,11 @@ static void fifo_push( lua_State* L, keeper_fifo* fifo, int _count) // expects exactly 1 value on the stack! // currently only called with a count of 1, but this may change in the future // function assumes that there is enough data in the fifo to satisfy the request -static void fifo_peek( lua_State* L, keeper_fifo* fifo, int _count) +static void fifo_peek( lua_State* L, keeper_fifo* fifo, int count_) { int i; - STACK_GROW( L, _count); - for( i = 0; i < _count; ++ i) + STACK_GROW( L, count_); + for( i = 0; i < count_; ++ i) { lua_rawgeti( L, 1, fifo->first + i); } @@ -384,54 +384,21 @@ int keepercall_limit( lua_State* L) return lua_gettop( L); } -//in: linda_ud key [val] +//in: linda_ud key [[val] ...] //out: true or nil int keepercall_set( lua_State* L) { bool_t should_wake_writers = FALSE; STACK_GROW( L, 6); - // make sure we have a value on the stack - if( lua_gettop( L) == 2) // ud key val? - { - lua_pushnil( L); // ud key nil - } // retrieve fifos associated with the linda - push_table( L, 1); // ud key val fifos - lua_replace( L, 1); // fifos key val + push_table( L, 1); // ud key [val [, ...]] fifos + lua_replace( L, 1); // fifos key [val [, ...]] - if( !lua_isnil( L, 3)) // set/replace contents stored at the specified key? + // make sure we have a value on the stack + if( lua_gettop( L) == 2) // fifos key { keeper_fifo* fifo; - lua_pushvalue( L, -2); // fifos key val key - lua_rawget( L, 1); // fifos key val fifo|nil - fifo = (keeper_fifo*) lua_touserdata( L, -1); - if( fifo == NULL) // can be NULL if we store a value at a new key - { // fifos key val nil - lua_pop( L, 1); // fifos key val - fifo_new( L); // fifos key val fifo - lua_pushvalue( L, 2); // fifos key val fifo key - lua_pushvalue( L, -2); // fifos key val fifo key fifo - lua_rawset( L, 1); // fifos key val fifo - } - else // the fifo exists, we just want to update its contents - { // fifos key val fifo - // we create room if the fifo was full but it is no longer the case - should_wake_writers = (fifo->limit > 0) && (fifo->count >= fifo->limit); - // empty the fifo for the specified key: replace uservalue with a virgin table, reset counters, but leave limit unchanged! - lua_newtable( L); // fifos key val fifo {} - lua_setuservalue( L, -2); // fifos key val fifo - fifo->first = 1; - fifo->count = 0; - } - fifo = prepare_fifo_access( L, -1); - lua_insert( L, -2); // fifos key fifo val - fifo_push( L, fifo, 1); // fifos key fifo - } - else // val == nil: we clear the key contents - { // fifos key nil - keeper_fifo* fifo; - lua_pop( L, 1); // fifos key lua_pushvalue( L, -1); // fifos key key lua_rawget( L, 1); // fifos key fifo|nil // empty the fifo for the specified key: replace uservalue with a virgin table, reset counters, but leave limit unchanged! @@ -456,13 +423,51 @@ int keepercall_set( lua_State* L) } } } + else // set/replace contents stored at the specified key? + { + int count = lua_gettop( L) - 2; // number of items we want to store + keeper_fifo* fifo; // fifos key [val [, ...]] + lua_pushvalue( L, 2); // fifos key [val [, ...]] key + lua_rawget( L, 1); // fifos key [val [, ...]] fifo|nil + fifo = (keeper_fifo*) lua_touserdata( L, -1); + if( fifo == NULL) // can be NULL if we store a value at a new key + { // fifos key [val [, ...]] nil + // no need to wake writers in that case, because a writer can't wait on an inexistent key + lua_pop( L, 1); // fifos key [val [, ...]] + fifo_new( L); // fifos key [val [, ...]] fifo + lua_pushvalue( L, 2); // fifos key [val [, ...]] fifo key + lua_pushvalue( L, -2); // fifos key [val [, ...]] fifo key fifo + lua_rawset( L, 1); // fifos key [val [, ...]] fifo + } + else // the fifo exists, we just want to update its contents + { // fifos key [val [, ...]] fifo + // we create room if the fifo was full but it is no longer the case + should_wake_writers = (fifo->limit > 0) && (fifo->count >= fifo->limit) && (count < fifo->limit); + // empty the fifo for the specified key: replace uservalue with a virgin table, reset counters, but leave limit unchanged! + lua_newtable( L); // fifos key [val [, ...]] fifo {} + lua_setuservalue( L, -2); // fifos key [val [, ...]] fifo + fifo->first = 1; + fifo->count = 0; + } + fifo = prepare_fifo_access( L, -1); + // move the fifo below the values we want to store + lua_insert( L, 3); // fifos key fifo [val [, ...]] + fifo_push( L, fifo, count); // fifos key fifo + } return should_wake_writers ? (lua_pushboolean( L, 1), 1) : 0; } -// in: linda_ud key +// in: linda_ud key [count] +// out: at most |