On Fri, May 25, 2012 at 01:57:04PM -0700, Joe Perches wrote: > On Fri, 2012-05-25 at 13:25 -0700, Kent Overstreet wrote: > > Asynchronous refcounty thingies; they embed a refcount and a work > > struct. Extensive documentation follows in include/linux/closure.h > [] > > diff --git a/include/linux/closure.h b/include/linux/closure.h > [] > > +enum closure_type { > > + TYPE_closure = 0, > > I still think these should be > CLOSURE_TYPE_closure > etc. Oh - yes, definitely. > > +#define __CLOSURE_TYPE(cl, _t) \ > > + __builtin_types_compatible_p(typeof(cl), struct _t) \ > > + ? TYPE_ ## _t : \ > CLOSURE_TYPE_##_t > > > +#define __closure_type(cl) \ > > +( \ > > + __CLOSURE_TYPE(cl, closure) \ > > + __CLOSURE_TYPE(cl, closure_with_waitlist) \ > > + __CLOSURE_TYPE(cl, closure_with_timer) \ > > + __CLOSURE_TYPE(cl, closure_with_waitlist_and_timer) \ > > + invalid_closure_type() \ > > +) > > You should still feel dirty about this... I feel a lot dirtier for implementing dynamic dispatch like this than the macro tricks. The macro stuff at least is pretty simple stuff that doesn't affect anything outside the 10 lines of code where it's used. > > +#define continue_at(_cl, _fn, _wq, ...) \ > > +do { \ > > + BUG_ON(!(_cl) || object_is_on_stack(_cl)); \ > > + closure_set_ip(_cl); \ > > + set_closure_fn(_cl, _fn, _wq); \ > > + closure_sub(_cl, CLOSURE_RUNNING + 1); \ > > + return __VA_ARGS__; \ > > +} while (0) > > Does this have to be a macro? Yes (though I could at least drop the __VA_ARGS__ part, I'm not using it anymore and it was always a hack). I explained why in the documentation, but basically the return is there to enforce correct usage (it can't prevent you from doing dumb things, but it can keep you from doing dumb things accidentally). It's not _just_ enforcing correct usage though, it leads to better and cleaner style. continue_at() is really flow control, so it makes the code clearer and easier to follow if that's how it's used (by using it also return). > > diff --git a/lib/closure.c b/lib/closure.c > [] > > +#define CL_FIELD(type, field) \ > > + case TYPE_ ## type: \ > > + return &container_of(cl, struct type, cl)->field > > + > > +static struct closure_waitlist *closure_waitlist(struct closure *cl) > > +{ > > + switch (cl->type) { > > + CL_FIELD(closure_with_waitlist, wait); > > + CL_FIELD(closure_with_waitlist_and_timer, wait); > > + default: > > + return NULL; > > + } > > +} > > Here: > > static struct closure_waitlist *closure_waitlist(struct closure *cl) > { > switch (cl->type) { > case CLOSURE_TYPE_closure_with_waitlist: > return &container_of(cl, struct closure_with_waitlist, cl)->wait; > case CLOSURE_TYPE_closure_with_waitlist_and_timer: > return &container_of(cl, struct closure_with_waitlist_and_timer, cl)->wait; > } > > return NULL; > } Alright, if you feel that strongly about it :) -- To unsubscribe from this list: send the line "unsubscribe linux-bcache" in the body of a message to majordomo@xxxxxxxxxxxxxxx More majordomo info at http://vger.kernel.org/majordomo-info.html