On Tue, 8 Dec 2009, Alan Stern wrote: > > > > So we actually _want_ the full rwlock semantics. > > I'm not convinced. Condense the description a little farther: > > Suspend: Children lock the parent first. When they are > finished they unlock the parent, allowing it to > proceed. > > Resume: Parent locks itself first. When it is finished > it unlocks itself, allowing the children to proceed. Yes. You can implement it with a simple lock with a count. Nobody debates that. But a simple counting lock _is_ a rwlock. Really. They are 100% semantically equivalent. There is no difference. > The whole readers vs. writers thing is a non-sequitur. No it's not. It's a 100% equivalent problem. It's purely a change of wording. The end result is the same. > The simplest generalization is to allow both multiple lockers and > multiple waiters. Call it a waitlock, for want of a better name: But we have that. It _has_ a better name: rwlocks. And the reason the name is better is because now the name describes all the semantics to anybody who has ever taken a course in operating systems or in parallelism. It's also a better implementation, because it actually _works_. > wait_lock(wl) > { > atomic_inc(&wl->count); > } > > wait_unlock(wl) > { > if (atomic_dec_and_test(&wl->count)) { > smp_mb__after_atomic_dec(); > wake_up_all(wl->wqh); > } > } > > wait_for_lock(wl) > { > wait_event(wl->wqh, atomic_read(&wl->count) == 0); > smp_rmb(); > } > > Note that both wait_lock() and wait_unlock() can be called > in_interrupt. And note how even though you sprinkled random memory barriers around, you still got it wrong. So you just implemented a buggy lock, and for what gain? Tell me exactly why your buggy lock (assuming you'd know enough about memory ordering to actually fix it) is better than just using the existing one? It's certainly not smaller. It's not faster. It doesn't have support for lockdep. And it's BUGGY. Really. Tell me why you want to re-implement an existing lock - badly. [ Hint: you need a smp_mb() *before* the atomic_dec() in wait-unlock, so that anybody else who sees the new value will be guaranteed to have seen anything else the unlocker did. You also need a smp_mb() in the wait_for_lock(), not a smp_rmb(). Can't allow writes to migrate up either. 'atomic_read()' does not imply any barriers. But most architectures can optimize these things for their particular memory ordering model, and do so in their rwsem implementation. ] > This becomes: > > usb_node_resume(node) > { > // Wait for parent to finish resume > wait_for_lock(node->parent->lock); > // .. before resuming outselves > node->resume(node) > > // Now we're all done > wait_unlock(node->lock); > } > > /* caller: */ > .. > // This can't block, because wait_lock() is non-blocking. > wait_lock(node->lock); > // Do the blocking part asynchronously > async_schedule(usb_node_resume, leaf); > .. Umm? Same thing, different words? That "wait_for_lock()" is equivalent to a 'read_lock()+read_unlock()'. We _could_ expose such a mechanism for rwsem's too, but why do it? It's actually nicer to use a real read-lock - and do it _around_ the operation, because now the locking also automatically gets things like overlapping suspends and resumes right. (Which you'd obviously hope never happens, but it's nice from a conceptual standpoint to know that the locking is robust). > Aren't waitlocks simpler than rwsems? Not as generally useful, > perhaps. But just as correct in this situation. NO! Dammit. I started this whole rant with this comment to Rafael: "The fact is, any time anybody makes up a new locking mechanism, THEY ALWAYS GET IT WRONG. Don't do it." Take heed. You got it wrong. Admit it. Locking is _hard_. SMP memory ordering is HARD. So leave locking to the pro's. They _also_ got it wrong, but they got it wrong several years ago, and fixed up their sh*t. This is why you use generic locking. ALWAYS. Linus _______________________________________________ linux-pm mailing list linux-pm@xxxxxxxxxxxxxxxxxxxxxxxxxx https://lists.linux-foundation.org/mailman/listinfo/linux-pm