On Tue, 2020-06-30 at 20:14 +0800, Xiubo Li wrote: > On 2020/6/30 19:29, Jeff Layton wrote: > > On Tue, 2020-06-30 at 03:52 -0400, xiubli@xxxxxxxxxx wrote: > > > From: Xiubo Li <xiubli@xxxxxxxxxx> > > > > > > This will send the caps/read/write/metadata metrics to any available > > > MDS only once per second as default, which will be the same as the > > > userland client. It will skip the MDS sessions which don't support > > > the metric collection, or the MDSs will close the socket connections > > > directly when it get an unknown type message. > > > > > [...] > > > > +static struct ceph_mds_session *metric_get_session(struct ceph_mds_client *mdsc) > > > +{ > > > + struct ceph_mds_session *s; > > > + int i; > > > + > > > + mutex_lock(&mdsc->mutex); > > > + for (i = 0; i < mdsc->max_sessions; i++) { > > > + s = __ceph_lookup_mds_session(mdsc, i); > > > + if (!s) > > > + continue; > > > + mutex_unlock(&mdsc->mutex); > > > + > > Why unlock here? AFAICT, it's safe to call ceph_put_mds_session with the > > mdsc->mutex held. > > Yeah, it is. Just wanted to make the critical section as small as > possible. And the following code no need the lock. > > Compared to the mutex lock acquisition is very expensive, we might not > benefit much from the smaller critical section. > > I will fix it. > Generally small critical sections are preferred, but almost all of these are in-memory operations. None of that should sleep, so we're almost certainly better off with less lock thrashing. If the lock isn't contended, then no harm is done. If it is, then we're better off not letting the cacheline bounce. > > > + /* > > > + * Skip it if MDS doesn't support the metric collection, > > > + * or the MDS will close the session's socket connection > > > + * directly when it get this message. > > > + */ > > > + if (check_session_state(s) && > > > + test_bit(CEPHFS_FEATURE_METRIC_COLLECT, &s->s_features)) { > > > + mdsc->metric.mds = i; > > > + return s; > > > + } > > > + ceph_put_mds_session(s); > > > + > > > + mutex_lock(&mdsc->mutex); > > > + } > > > + mutex_unlock(&mdsc->mutex); > > > + > > > + return NULL; > > > +} > > > + > > > +static void metric_delayed_work(struct work_struct *work) > > > +{ > > > + struct ceph_client_metric *m = > > > + container_of(work, struct ceph_client_metric, delayed_work.work); > > > + struct ceph_mds_client *mdsc = > > > + container_of(m, struct ceph_mds_client, metric); > > > + struct ceph_mds_session *s = NULL; > > > + u64 nr_caps = atomic64_read(&m->total_caps); > > > + > > > + /* No mds supports the metric collection, will stop the work */ > > > + if (!atomic_read(&m->mds_cnt)) > > > + return; > > > + > > > + mutex_lock(&mdsc->mutex); > > > + s = __ceph_lookup_mds_session(mdsc, m->mds); > > > + mutex_unlock(&mdsc->mutex); > > > > Instead of doing a lookup of the mds every time we need to do this, > > would it be better to instead just do a lookup before you first schedule > > the work and keep a reference to it until the session state is no longer > > good? > > > > With that, you'd only need to take the mutex here if check_session_state > > indicated that the session you had saved was no longer good. > > This sounds very cool and with this we can get rid of the mutex lock in > normal case. > Yeah. I think the code could be simplified the code in other ways too. You have per-mdsc work now, so there's no problem scheduling the work more than once. You can just schedule it any time you get a new session that has the feature flag set. Keep a metrics session pointer in the mdsc->metric, and start with it set to NULL. When the work runs, do a lookup if the pointer is NULL or if the current one isn't valid any more. At the end, only reschedule the work if we found a suitable session. That should eliminate the need for the mdsc.metric->mds_cnt counter. > > > + if (unlikely(!s || !check_session_state(s) || > > > + !test_bit(CEPHFS_FEATURE_METRIC_COLLECT, &s->s_features))) > > > + s = metric_get_session(mdsc); > > > + > > If we do need to keep doing a lookup every time, then it'd probably be > > better to do the above check while holding the mdsc->mutex and just have > > metric_get_session expect to be called with the mutex already held. > > > > FWIW, mutexes are expensive locks since you can end up having to > > schedule() if you can't get it. Minimizing the number of acquisitions > > and simply holding them for a little longer is often the more efficient > > approach. > > Okay, will do that. > > [...] > > > > > > > diff --git a/fs/ceph/super.c b/fs/ceph/super.c > > > index c9784eb1..cd33836 100644 > > > --- a/fs/ceph/super.c > > > +++ b/fs/ceph/super.c > > > @@ -27,6 +27,9 @@ > > > #include <linux/ceph/auth.h> > > > #include <linux/ceph/debugfs.h> > > > > > > +static DEFINE_MUTEX(ceph_fsc_lock); > > I think this could be a spinlock. None of the operations it protects > > look like they can sleep. > > Will fix it. > > Thanks. > > -- Jeff Layton <jlayton@xxxxxxxxxx>