NAME IPC::ScoreBoard - IPC similar to the apache scoreboard SYNOPSIS use IPC::ScoreBoard; # create an anonymous scoreboard my $sb=SB::anon $nslots, $slotsize, $extra; # create a file base board my $sb=SB::named $filename, $nslots, $slotsize, $extra; # open a file based board my ($sb, $nslots, $slotsize, $extra)=SB::open $filename; # set/set a value SB::set $sb, $slotidx, $elidx, $integer_value; $value=SB::get $sb, $slotidx, $elidx; @values=SB::get_all $sb, $slotidx; # increment/decrement SB::incr $sb, $slotidx, $elidx, $integer_value; SB::decr $sb, $slotidx, $elidx, $integer_value; # sum functions $sum=SB::sum $sb, $elidx; @sums=SB::sum_all $sb; # access extra space SB::set_extra $sb, $elidx, $integer_value; $value=SB::get_extra $sb, $elidx; @values=SB::get_all_extra $sb; SB::incr_extra $sb, $elidx, $integer_value; SB::decr_extra $sb, $elidx, $integer_value; # fetch parameters $nslots=SB::nslots $sb; $slotsize=SB::slotsize $sb; $nextra=SB::nextra $sb; # does the compiler provide atomic increment/decrement operations? if( SB::have_atomics ) { # increment and decrement operations are atomic } INSTALLATION perl Makefile.PL make make test make install DEPENDENCIES * perl 5.8.8 * File::Map 0.21 DESCRIPTION A scoreboard is a set of integer numbers residing in shared memory. It is organized as 2-dimensional array where a line in one of the dimensions is called a slot. So, in other words the scoreboard is a set of slots and each slot is a set of integer numbers. The idea is that in a system of processes or threads of execution each process *owns* a slot. A process can change the values in its own slot at will but must adhere to read-only access to other slots. There is one extra slot at the end of the scoreboard that is allowed to be used by every process. However this module does not provide any kind of locking to synchronize access. The extra slot can differ in size from the other normal slots. A scoreboard can be anonymous or it can have a name in the file system and hence be accessed by unrelated processes. What is that good for? Suppose a system of processes that handle certain requests. Now, you want to implement a monitor that shows the overall number of requests handled so far by the system as a whole. One way to do that is to use a shared variable that is incremented each time a process has finished a request. But access to this variable has to be synchronized by some type of locking. Otherwise 2 or even more processes can read the shared variable at the same time. Then each of them increments its own value and writes it back. In the end only the value written by the last process hits the memory. All other increments are lost. A lock free way could be to have each process increment its own variable. Then the monitor would have to sum up all the variables of the processes. But the system is now lock-free. USAGE A scoreboard object is a reference to a scalar. Its methods can be invoked in the usual object style, "$sb->get(42,19)", or as subroutines, "IPC::ScoreBoard::get $sb, 42, 19". The latter variant is a bit faster but involves a lot of typing. To mitigate that all functions are exported to the "SB::" namespace if the module is included via "use". If it is included via "require" nothing is exported. Neither is it if the parameter ":noshortcuts" is passed to "use": use IPC::ScoreBoard; # generates shortcuts SB::get & co. use IPC::ScoreBoard (); # no shortcuts use IPC::ScoreBoard ':noshortcuts'; # no shortcuts require IPC::ScoreBoard; # no shortcuts All data access functions throw an exception if access outside the boundaries of the slot or scoreboard is tried. The following section shows only the shortcut usage. Remember all functions can also be called as $object_or_class->functionname(@param); or as IPC::ScoreBoard::functionname $scoreboard, @param; Scoreboard creation SB::anon $nslots, $slotsize, $nextra creates an anonymous scoreboard with space for $nslots slots and $slotsize "IV" values per slot. The extra slot contains $nextra "IV" values. "anon" returns the scoreboard object. In case of an error an exception is thrown. Example: my $sb=IPC::ScoreBoard->anon($nslots, $slotsize, $nextra); my $sb=IPC::ScoreBoard::anon $nslots, $slotsize, $nextra; my $sb=SB::anon $nslots, $slotsize, $nextra; SB::named $filename, $nslots, $slotsize, $nextra similar to "anon" but creates a named scoreboard with the name $filename. Example: my $sb=IPC::ScoreBoard->named($filename, $nslots, $slotsize, $nextra); my $sb=IPC::ScoreBoard::named $filename, $nslots, $slotsize, $nextra; my $sb=SB::named $filename, $nslots, $slotsize, $nextra; SB::open $filename similar to "anon" but connects to or opens an existing named scoreboard with the name $filename. Besides the scoreboard object the scoreboard parameters $nslots, $slotsize, $nextra are returned: Example: my ($sb, $nslots, $slotsize, $extra)=IPC::ScoreBoard->open($filename); my ($sb, $nslots, $slotsize, $extra)=IPC::ScoreBoard::open $filename; my ($sb, $nslots, $slotsize, $extra)=SB::open $filename; Data manipulation SB::set $sb, $slotidx, $elidx, $value; sets the $elidxth (counting from 0) element in slot number $slotidx (also counting from 0) to $value. $value is interpreted as integer. The new value is returned. SB::get $sb, $slotidx, $elidx; reads the value at position $elidx in slot number $slotidx. SB::incr $sb, $slotidx, $elidx, $amount; SB::decr $sb, $slotidx, $elidx, $amount; these 2 functions increment or decrement the value at position $elidx in slot number $slotidx. $amount is optional. If ommitted 1 is used. If supported by the compiler atomic operations are used to do that. That means, even if multiple processes increment or decrement a certain value in parallel nothing is lost as described in the DESCRIPTION. The new value is returned. SB::sum $sb, $elidx; sums up the values at a position $elidx over all slots (except for the extra one). SB::get_all $sb, $slotidx; returns a list of all values of slot number $slotidx. SB::sum_all $sb; returns a list of sums. The equivalent in perl could read: @sums=map { SB::sum $sb, $_ } 0..$slotsize; SB::set_extra $sb, $elidx, $value; sets the value at position $elidx in the extra slot. SB::get_extra $sb, $elidx; reads the value at position $elidx in the extra slot. SB::incr_extra $sb, $elidx, $amount; SB::decr_extra $sb, $elidx, $amount; these 2 functions increment or decrement the value at position $elidx in the extra slot. $amount is optional. If ommitted 1 is used. If supported by the compiler atomic operations are used to do that. That means, even if multiple processes increment or decrement a certain value in parallel nothing is lost as described in the DESCRIPTION. The new value is returned. SB::get_all_extra $sb; returns the list of all values from the extra slot. Auxiliary functions SB::nslots $sb returns the number of slots in the scoreboard SB::slotsize $sb returns the number of "IV"s in each slot SB::nextra $sb returns the number of "IV"s in the extra slot SB::offset_of $sb, $slotidx, $elidx converts a slotnumber and an index within the slot into a byte offset from the beginning of the scoreboard. If both $slotidx and $elidx are given the offset of the slot element is returned. If $elidx is ommitted or undefined $slotidx is taken as an element index within the extra slot: $sb->offset_of(2, 3); # 3rd IV in 2nd slot SB::offset_of($sb, 3); # 3rd IV of extra slot This allows to store data other than integers. Example: # store "hugo" in extra[2..4] substr($$sb, $sb->offset_of(2), 3*$ivlen, pack( "Z".(3*$ivlen), "hugo")); # retrieve "hugo" (unpack "x".$sb->offset_of(2)."Z*", $$sb)[0] Notes on the example: * Make sure the replacement string has exactly the length as given in the 3rd "substr" parameter. * Don't use the lvalue form of "substr". It stores a reference to $$sb and hence if $$sb goes out of scope the scoreboard won't be unmapped. Don't do: substr($$sb, $sb->offset_of(2), 3*$ivlen)=pack( "Z".(3*$ivlen), "hugo"); * Remember, the "Z" pack format stores always a "NULL" byte at the end of the string. So, the example works because "hugo" does not contain a "NULL" byte. If you need to store binary data a more sophisticated format as "W/a" could be used. But watch out for the maximum length. SB::have_atomics returns true if "IPC::ScoreBoard" has been compiled with a compiler that supports atomic increment/decrement operations. EXPORT Nothing. SEE ALSO * for more information about atomic operations * File::Map the underlying memory mapper AUTHOR Torsten Förtsch, COPYRIGHT AND LICENSE Copyright (C) 2010 by Torsten Förtsch This library is free software; you can redistribute it and/or modify it under the same terms as Perl itself, either Perl version 5.10.0 or, at your option, any later version of Perl 5 you may have available.