From owner-freebsd-arch@FreeBSD.ORG Sun Aug 22 19:05:50 2010 Return-Path: Delivered-To: freebsd-arch@freebsd.org Received: from mx1.freebsd.org (mx1.freebsd.org [IPv6:2001:4f8:fff6::34]) by hub.freebsd.org (Postfix) with ESMTP id 2518010656A5; Sun, 22 Aug 2010 19:05:50 +0000 (UTC) (envelope-from max@love2party.net) Received: from moutng.kundenserver.de (moutng.kundenserver.de [212.227.126.187]) by mx1.freebsd.org (Postfix) with ESMTP id C12AC8FC1F; Sun, 22 Aug 2010 19:05:49 +0000 (UTC) Received: from f8x64.laiers.local (dslb-088-066-049-083.pools.arcor-ip.net [88.66.49.83]) by mrelayeu.kundenserver.de (node=mreu2) with ESMTP (Nemesis) id 0Mao5W-1OYLdm02ws-00JuQt; Sun, 22 Aug 2010 21:05:48 +0200 From: Max Laier Organization: FreeBSD To: Stephan Uphoff Date: Sun, 22 Aug 2010 21:05:47 +0200 User-Agent: KMail/1.13.5 (FreeBSD/8.1-RELEASE; KDE/4.4.5; amd64; ; ) References: <201008160515.21412.max@love2party.net> <4C7042BA.8000402@freebsd.org> In-Reply-To: <4C7042BA.8000402@freebsd.org> MIME-Version: 1.0 Content-Type: Text/Plain; charset="iso-8859-1" Content-Transfer-Encoding: 7bit Message-Id: <201008222105.47276.max@love2party.net> X-Provags-ID: V02:K0:YNor0iNpkdfjelCFoSMWyZtpXJhZ7ohG2c5/aghqCgx Y0kFjQW7ZEyXAxH3BAGR4CG/+A+ExZsSiYd7FNRUFc2gSxOhnj 9Vte5ZrKUUlT5A+foYJ0cZDmo3AbZpHypBmITXvyePacesCCag vJMxDHLlrIfSRo1pPmWYt4Q0KgcKqgeDxRzSKKX/G+r7VQDzFj n9xX7r3jji5syyUh2awzQ== Cc: freebsd-arch@freebsd.org Subject: Re: rmlock(9) two additions X-BeenThere: freebsd-arch@freebsd.org X-Mailman-Version: 2.1.5 Precedence: list List-Id: Discussion related to FreeBSD architecture List-Unsubscribe: , List-Archive: List-Post: List-Help: List-Subscribe: , X-List-Received-Date: Sun, 22 Aug 2010 19:05:50 -0000 On Saturday 21 August 2010 23:18:50 Stephan Uphoff wrote: > Max Laier wrote: > > Hi, > > > > I'd like to run two additions to rmlock(9) by you: > > > > 1) See the attached patch. It adds rm_try_rlock() which avoids taking > > the mutex if the lock is currently in a write episode. The only > > overhead to the hot path is an additional argument and a return value > > for _rm_rlock*. If you are worried about that, it can obviously be done > > in a separate code path, but I reckon it not worth the code crunch. > > Finally, there is one additional branch to check the "trylock" argument, > > but that's well past the hot path. > > The rm_try_rlock() will never succeed when the lock was last locked as > writer. > Maybe add: > > void > _rm_wunlock(struct rmlock *rm) > { > + rm->rm_noreadtoken = 0; > mtx_unlock(&rm->rm_lock); > } > > But then > > _rm_wlock(struct rmlock *rm) > > always needs to use IPIs - even when the lock was used last as a write > lock. I don't think this is a big problem - I can't see many use cases for rmlocks where you'd routinely see repeated wlocks without rlocks between them. However, I think there should be a release memory barrier before/while clearing rm_noreadtoken, otherwise readers may not see the data writes that are supposed to be protected by the lock?!? > Alternatively something like: > > if (trylock) { > if(mtx_trylock( &rm->rm_lock) == 0) > return (0); > } > else > { > mtx_lock(&rm->rm_lock); > } > > would work - but has a race. Two readers colliding just after a writer > (with the second not succeeding in trylocking the mutex) leads to not > granting the read lock (also it would be possible to do so). Also not too much of a problem, in my opinion. There is no time order between read locks and thus it is okay to grant one and fail another - eventhough they arrive "at the same time". A caller to trylock must always accept failure (unless it's a recursive use - and this is handled). > Let me think about it a bit. I believe either solution will work. #1 is a bit more in the spirit of the rmlock - i.e. make the read case cheap and the write case expensive. I'm just not sure about the lock semantics. I guess a atomic_store_rel_int(&rm->rm_noreadtoken, 0); should work. > > 2) No code for this yet - testing the waters first. I'd like to add the > > ability to replace the mutex for writer synchronization with a general > > lock - especially to be able to use a sx(9) lock here. > > > > The reason for #2 is the following use case in a debugging facility: > > "reader": > > if (rm_try_rlock()) { > > > > grab_per_cpu_buffer(); > > fill_per_cpu_buffer(); > > rm_runlock(); > > > > } > > > > "writer" - better exclusive access thread: > > rm_wlock(); > > collect_buffers_and_copy_out_to_userspace(); > > rm_wunlock(); > > > > This is much cleaner and possibly cheaper than the various hand rolled > > versions I've come across, that try to get the same synchronization with > > atomic operations. If we could sleep with the wlock held, we can also > > avoid copying the buffer contents, or swapping buffers. > > > > Is there any concern about either of this? Any objection? Input? > > Will take a look at your second patch soonish. > > Just ask per IPI for a copy of per cpu buffers (but not a copy to user > space) - and delay the copy when an update is in progress? Think huge circular per cpu buffer that are filled at high rates. Of course we could allocate new buffers and swap out while locked, but since this is a debugging facility it is better to miss a few events while copying out, rather than spending twice the memory. Thanks, Max