On Mon, Dec 02, 2013 at 03:19:45PM +0400, Vladimir Davydov wrote: > FS-shrinkers, which shrink dcaches and icaches, keep dentries and inodes > in list_lru structures in order to evict least recently used objects. > With per-memcg kmem shrinking infrastructure introduced, we have to make > those LRU lists per-memcg in order to allow shrinking FS caches that > belong to different memory cgroups independently. > > This patch addresses the issue by introducing struct memcg_list_lru. > This struct aggregates list_lru objects for each kmem-active memcg, and > keeps it uptodate whenever a memcg is created or destroyed. Its > interface is very simple: it only allows to get the pointer to the > appropriate list_lru object from a memcg or a kmem ptr, which should be > further operated with conventional list_lru methods. Basically The idea was that the memcg LRUs hide entirely behind the generic list_lru interface so that any cache that used the list_lru insfrastructure got memcg capabilities for free. memcg's to shrink were to be passed through the shrinker control shrinkers to the list LRU code, and it then did all the "which lru are we using" logic internally. What you've done is driven all the "which LRU are we using" logic into every single caller location. i.e. you've just broken the underlying design principle that Glauber and I had worked towards with this code - that memcg aware LRUs should be completely transparent to list_lru users. Just like NUMA awareness came for free with the list_lru code, so should memcg awareness.... > +/* > + * The following structure can be used to reclaim kmem objects accounted to > + * different memory cgroups independently. It aggregates a set of list_lru > + * objects, one for each kmem-enabled memcg, and provides the method to get > + * the lru corresponding to a memcg. > + */ > +struct memcg_list_lru { > + struct list_lru global_lru; > + > +#ifdef CONFIG_MEMCG_KMEM > + struct list_lru **memcg_lrus; /* rcu-protected array of per-memcg > + lrus, indexed by memcg_cache_id() */ > + > + struct list_head list; /* list of all memcg-aware lrus */ > + > + /* > + * The memcg_lrus array is rcu protected, so we can only free it after > + * a call to synchronize_rcu(). To avoid multiple calls to > + * synchronize_rcu() when many lrus get updated at the same time, which > + * is a typical scenario, we will store the pointer to the previous > + * version of the array in the old_lrus variable for each lru, and then > + * free them all at once after a single call to synchronize_rcu(). > + */ > + void *old_lrus; > +#endif > +}; Really, this should be embedded in the struct list_lru, not wrapping around the outside. I don't see any changelog to tell me why you changed the code from what was last in Glauber's tree, so can you explain why exposing all this memcg stuff to everyone is a good idea? Cheers, Dave. -- Dave Chinner david@xxxxxxxxxxxxx -- To unsubscribe, send a message with 'unsubscribe linux-mm' in the body to majordomo@xxxxxxxxx. For more info on Linux MM, see: http://www.linux-mm.org/ . Don't email: <a href=mailto:"dont@xxxxxxxxx"> email@xxxxxxxxx </a>