172445076SMariusz Zaborski.\" 272445076SMariusz Zaborski.\" Copyright (C) 2019 Mariusz Zaborski <oshogbo@FreeBSD.org> 372445076SMariusz Zaborski.\" 472445076SMariusz Zaborski.\" Redistribution and use in source and binary forms, with or without 572445076SMariusz Zaborski.\" modification, are permitted provided that the following conditions 672445076SMariusz Zaborski.\" are met: 772445076SMariusz Zaborski.\" 1. Redistributions of source code must retain the above copyright 872445076SMariusz Zaborski.\" notice(s), this list of conditions and the following disclaimer as 972445076SMariusz Zaborski.\" the first lines of this file unmodified other than the possible 1072445076SMariusz Zaborski.\" addition of one or more copyright notices. 1172445076SMariusz Zaborski.\" 2. Redistributions in binary form must reproduce the above copyright 1272445076SMariusz Zaborski.\" notice(s), this list of conditions and the following disclaimer in the 1372445076SMariusz Zaborski.\" documentation and/or other materials provided with the distribution. 1472445076SMariusz Zaborski.\" 1572445076SMariusz Zaborski.\" THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDER(S) ``AS IS'' AND ANY 1672445076SMariusz Zaborski.\" EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 1772445076SMariusz Zaborski.\" WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE 1872445076SMariusz Zaborski.\" DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER(S) BE LIABLE FOR ANY 1972445076SMariusz Zaborski.\" DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES 2072445076SMariusz Zaborski.\" (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR 2172445076SMariusz Zaborski.\" SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER 2272445076SMariusz Zaborski.\" CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 2372445076SMariusz Zaborski.\" LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 2472445076SMariusz Zaborski.\" OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH 2572445076SMariusz Zaborski.\" DAMAGE. 2672445076SMariusz Zaborski.\" 2772445076SMariusz Zaborski.Dd July 29, 2019 2872445076SMariusz Zaborski.Dt SEQC 9 2972445076SMariusz Zaborski.Os 3072445076SMariusz Zaborski.Sh NAME 3172445076SMariusz Zaborski.Nm seqc_consistent , 3272445076SMariusz Zaborski.Nm seqc_read , 3372445076SMariusz Zaborski.Nm seqc_write_begin , 3472445076SMariusz Zaborski.Nm seqc_write_end 3572445076SMariusz Zaborski.Nd "lockless read algorithm" 3672445076SMariusz Zaborski.Sh SYNOPSIS 3772445076SMariusz Zaborski.In sys/seqc.h 3872445076SMariusz Zaborski.Ft void 3972445076SMariusz Zaborski.Fn seqc_write_begin "seqc_t *seqcp" 4072445076SMariusz Zaborski.Ft void 4172445076SMariusz Zaborski.Fn seqc_write_end "seqc_t *seqcp" 4272445076SMariusz Zaborski.Ft seqc_t 4372445076SMariusz Zaborski.Fn seqc_read "seqc_t *seqcp" 4472445076SMariusz Zaborski.Ft seqc_t 4572445076SMariusz Zaborski.Fn seqc_consistent "const seqc_t *seqcp" "seqc_t oldseqc" 4672445076SMariusz Zaborski.Sh DESCRIPTION 4772445076SMariusz ZaborskiThe 4872445076SMariusz Zaborski.Nm seqc 4972445076SMariusz Zaborskiallows zero or more readers and zero or one writer to concurrently access 5072445076SMariusz Zaborskian object, providing a consistent snapshot of the object for readers. 5172445076SMariusz ZaborskiNo mutual exclusion between readers and writers is required, 5272445076SMariusz Zaborskibut readers may be starved indefinitely by writers. 5372445076SMariusz Zaborski.Pp 5472445076SMariusz ZaborskiThe functions 5572445076SMariusz Zaborski.Fn seqc_write_begin 5672445076SMariusz Zaborskiand 5772445076SMariusz Zaborski.Fn seqc_write_end 5872445076SMariusz Zaborskiare used to create a transaction for writer, and notify the readers that the 5972445076SMariusz Zaborskiobject will be modified. 6072445076SMariusz Zaborski.Pp 6172445076SMariusz ZaborskiThe 6272445076SMariusz Zaborski.Fn seqc_read 6372445076SMariusz Zaborskifunction returns the current sequence number. 6472445076SMariusz ZaborskiIf a writer has started a transaction, this function will spin until the 6572445076SMariusz Zaborskitransaction has ended. 6672445076SMariusz Zaborski.Pp 6772445076SMariusz ZaborskiThe 6872445076SMariusz Zaborski.Fn seqc_consistent 6972445076SMariusz Zaborskifunction compares the sequence number with a previously fetched value. 7072445076SMariusz ZaborskiThe 7172445076SMariusz Zaborski.Fa oldseqc 7272445076SMariusz Zaborskivariable should contain a sequence number from the beginning of read 7372445076SMariusz Zaborskitransaction. 7472445076SMariusz Zaborski.Pp 7572445076SMariusz ZaborskiThe reader at the end of a transaction checks if the sequence number has 7672445076SMariusz Zaborskichanged. 7772445076SMariusz ZaborskiIf the sequence number didn't change the object wasn't modified, and fetched 7872445076SMariusz Zaborskivariables are valid. 7972445076SMariusz ZaborskiIf the sequence number changed the object was modified and the fetch should be 8072445076SMariusz Zaborskirepeated. 8172445076SMariusz ZaborskiIn case when sequence number is odd the object change is in progress and the 8272445076SMariusz Zaborskireader will wait until the write will the sequence number will become even. 8372445076SMariusz Zaborski.Sh EXAMPLES 8472445076SMariusz ZaborskiThe following example for a writer changees the 8572445076SMariusz Zaborski.Va var1 8672445076SMariusz Zaborskiand 8772445076SMariusz Zaborski.Va var2 8872445076SMariusz Zaborskivariables in the 8972445076SMariusz Zaborski.Va obj 9072445076SMariusz Zaborskistructure: 9172445076SMariusz Zaborski.Bd -literal 9272445076SMariusz Zaborskilock_exclusive(&obj->lock); 9372445076SMariusz Zaborskiseqc_write_begin(&obj->seqc); 9472445076SMariusz Zaborskiobj->var1 = 1; 9572445076SMariusz Zaborskiobj->var2 = 2; 9672445076SMariusz Zaborskiseqc_write_end(&obj->seqc); 9772445076SMariusz Zaborskiunlock_exclusive(&obj->lock); 9872445076SMariusz Zaborski.Ed 99*da342999SLi-Wen Hsu.Pp 10072445076SMariusz ZaborskiThe following example for a reader reads the 10172445076SMariusz Zaborski.Va var1 10272445076SMariusz Zaborskiand 10372445076SMariusz Zaborski.Va var2 10472445076SMariusz Zaborskivariables from the 10572445076SMariusz Zaborski.Va obj 10672445076SMariusz Zaborskistructure. 10772445076SMariusz ZaborskiIn the case where the sequence number was changed it restarts the whole process. 10872445076SMariusz Zaborski.Bd -literal 10972445076SMariusz Zaborskiint var1, var2; 11072445076SMariusz Zaborskiseqc_t seqc; 11172445076SMariusz Zaborski 11272445076SMariusz Zaborskifor (;;) { 11372445076SMariusz Zaborski seqc = seqc_read(&obj->seqc); 11472445076SMariusz Zaborski var1 = obj->var1; 11572445076SMariusz Zaborski var2 = obj->var2; 11672445076SMariusz Zaborski if (seqc_consistent(&obj->seqc, seqc)) 11772445076SMariusz Zaborski break; 11872445076SMariusz Zaborski} 11972445076SMariusz Zaborski.Ed 12072445076SMariusz Zaborski.Sh AUTHORS 12172445076SMariusz ZaborskiThe 12272445076SMariusz Zaborski.Nm seqc 12372445076SMariusz Zaborskifunctions was implemented by 12472445076SMariusz Zaborski.An Mateusz Guzik Aq Mt mjg@FreeBSD.org . 12572445076SMariusz ZaborskiThis manual page was written by 12672445076SMariusz Zaborski.An Mariusz Zaborski Aq Mt oshogbo@FreeBSD.org . 12772445076SMariusz Zaborski.Sh CAVEATS 12872445076SMariusz ZaborskiThere is no guarantee of progress for readers. 12972445076SMariusz ZaborskiIn case when there are a lot of writers the reader can be starved. 13072445076SMariusz ZaborskiThis concern may be solved by returning error after a few attempts. 13172445076SMariusz Zaborski.Pp 13272445076SMariusz ZaborskiTheoretically if reading takes a very long time, and when there are many writers 13372445076SMariusz Zaborskithe counter may overflow and wrap around to the same value. 13472445076SMariusz ZaborskiIn that case the reader will not notice that the object was changed. 13572445076SMariusz ZaborskiGiven that this needs 4 billion transactional writes across a single contended 13672445076SMariusz Zaborskireader, it is unlikely to ever happen. 13772445076SMariusz ZaborskiThis could be avoided by extending the interface to allow 64-bit counters. 138