[Devel] [PATCH rh7 04/14] Port diff-vznetstat-support-of-resetting-of-traffic-statistics-on-running-CTs
Vladimir Davydov
vdavydov at parallels.com
Tue Jun 23 09:29:40 PDT 2015
Author: Kirill Tkhai
Email: ktkhai at parallels.com
Subject: vznetstat: support of resetting of traffic statistics on
Date: Thu, 13 Mar 2014 12:47:57 +0400
This adds a possibility of clearing VE's venet_stat when container
is running. Currently, to do that user needs to reboot the CT and
to complitely destroy statistics.
Patch implements two new ioctls: VZCTL_TC_CLEAR_STAT and
VZCTL_TC_CLEAR_ALL_STAT (clear the VE's statistics and clear
all available statistics).
We schedule clearing work using schedule_on_each_cpu(). This
function is not exported, so the patch does that. Scheduled
work is executed under disabled smp hotplug, so we do not
need in additional export of {get,put}_online_cpus(),
which are GPL-only, to clear statistics on temporary down
CPUs. Otherwise we'd have to export them (at least something,
because kernel does not allow not GPL modules to control
hotplug).
Also I add __percpu to "struct venet_stat" description,
because it's percpu.
More info in the JIRA request https://jira.sw.ru/browse/PSBM-21243
Signed-off-by: Kirill Tkhai <ktkhai at parallels.com>
Acked-by: Vladimir Davydov <vdavydov at parallels.com>
=============================================================================
Author: Kirill Tkhai
Email: ktkhai at parallels.com
Subject: workqueue: export schedule_on_each_cpu() for vznetstat
Date: Thu, 13 Mar 2014 12:47:57 +0400
This is a subsidiary patch which just export a function for the following patch:
diff-vznetstat-support-of-resetting-of-traffic-statistics-on-running-CTs
This adds a possibility of clearing VE's venet_stat when container
is running. Currently, to do that user needs to reboot the CT and
to complitely destroy statistics.
Patch implements two new ioctls: VZCTL_TC_CLEAR_STAT and
VZCTL_TC_CLEAR_ALL_STAT (clear the VE's statistics and clear
all available statistics).
We schedule clearing work using schedule_on_each_cpu(). This
function is not exported, so the patch does that. Scheduled
work is executed under disabled smp hotplug, so we do not
need in additional export of {get,put}_online_cpus(),
which are GPL-only, to clear statistics on temporary down
CPUs. Otherwise we'd have to export them (at least something,
because kernel does not allow not GPL modules to control
hotplug).
Also I add __percpu to "struct venet_stat" description,
because it's percpu.
More info in the JIRA request https://jira.sw.ru/browse/PSBM-21243
Signed-off-by: Kirill Tkhai <ktkhai at parallels.com>
Acked-by: Vladimir Davydov <vdavydov at parallels.com>
=============================================================================
Related to https://jira.sw.ru/browse/PSBM-33650
Signed-off-by: Vladimir Davydov <vdavydov at parallels.com>
---
include/linux/vznetstat.h | 4 +-
include/uapi/linux/vzctl_netstat.h | 3 ++
kernel/ve/vznetstat/vznetstat.c | 107 +++++++++++++++++++++++++++++++++++++
kernel/workqueue.c | 1 +
4 files changed, 113 insertions(+), 2 deletions(-)
diff --git a/include/linux/vznetstat.h b/include/linux/vznetstat.h
index b6627cc3f6f6..11b8419e057b 100644
--- a/include/linux/vznetstat.h
+++ b/include/linux/vznetstat.h
@@ -38,8 +38,8 @@ struct venet_stat {
unsigned long flags;
atomic_t users;
- struct acct_stat *ipv4_stat;
- struct acct_stat *ipv6_stat;
+ struct acct_stat __percpu *ipv4_stat;
+ struct acct_stat __percpu *ipv6_stat;
};
static inline int venet_acct_skb_size(struct sk_buff *skb)
diff --git a/include/uapi/linux/vzctl_netstat.h b/include/uapi/linux/vzctl_netstat.h
index 124c33811d65..b81435e758f0 100644
--- a/include/uapi/linux/vzctl_netstat.h
+++ b/include/uapi/linux/vzctl_netstat.h
@@ -83,6 +83,9 @@ struct vzctl_tc_set_base {
#define VZCTL_TC_CLASS_NUM_V6 _IO(VZTCCTLTYPE, 16)
+#define VZCTL_TC_CLEAR_STAT _IO(VZTCCTLTYPE, 17)
+#define VZCTL_TC_CLEAR_ALL_STAT _IO(VZTCCTLTYPE, 18)
+
#ifdef __KERNEL__
#ifdef CONFIG_COMPAT
#include <linux/compat.h>
diff --git a/kernel/ve/vznetstat/vznetstat.c b/kernel/ve/vznetstat/vznetstat.c
index dcdd85e69a6a..2f97b2c3e4be 100644
--- a/kernel/ve/vznetstat/vznetstat.c
+++ b/kernel/ve/vznetstat/vznetstat.c
@@ -398,6 +398,106 @@ static void venet_acct_destroy_all_stat(void)
write_unlock_irq(&tc_lock);
}
+static DEFINE_MUTEX(req_mutex);
+static struct venet_stat *req_stat;
+
+static void zero_venet_stat(struct venet_stat *stat, unsigned cpu)
+{
+ struct acct_stat *acct;
+
+ acct = per_cpu_ptr(stat->ipv4_stat, cpu);
+ memset(acct, 0, sizeof(*acct));
+ acct = per_cpu_ptr(stat->ipv6_stat, cpu);
+ memset(acct, 0, sizeof(*acct));
+}
+
+static void clear_one_percpu_statistics(struct work_struct *dummy)
+{
+ unsigned cpu, this_cpu = get_cpu();
+
+ zero_venet_stat(req_stat, this_cpu);
+
+ if (cpumask_first(cpu_online_mask) != this_cpu)
+ goto out;
+
+ /* First cpu clears statistics on all offline cpus */
+ for_each_possible_cpu(cpu)
+ if (!cpu_online(cpu))
+ zero_venet_stat(req_stat, cpu);
+out:
+ put_cpu();
+}
+
+/* Clear VE's statistics */
+static int venet_acct_clear_stat(envid_t veid)
+{
+ int ret = -EINTR;
+
+ if (mutex_lock_interruptible(&req_mutex))
+ goto out;
+
+ req_stat = venet_acct_find_stat(veid);
+ if (!req_stat) {
+ ret = -ESRCH;
+ goto unlock;
+ }
+
+ ret = schedule_on_each_cpu(clear_one_percpu_statistics);
+
+ venet_acct_put_stat(req_stat);
+unlock:
+ mutex_unlock(&req_mutex);
+out:
+ return ret;
+}
+
+static void clear_all_percpu_statistics(struct work_struct *dummy)
+{
+ unsigned cpu, this_cpu = smp_processor_id();
+ struct venet_stat *stat = NULL;
+ int other = 0, hash = 0;
+
+ /*
+ * Some cpus may be offline, and schedule_on_each_cpu()
+ * does not create a work on them.
+ * Work on the first online CPU clears their statistics.
+ * Hotplug is disabled by schedule_on_each_cpu().
+ */
+ if (cpumask_first(cpu_online_mask) == this_cpu)
+ other = 1;
+
+ read_lock(&tc_lock);
+
+ while ((stat = next_stat(&hash, stat)) != NULL) {
+ zero_venet_stat(stat, this_cpu);
+
+ if (!other)
+ continue;
+
+ /* Clear statistics on not active cpus */
+ for_each_possible_cpu(cpu)
+ if (!cpu_online(cpu))
+ zero_venet_stat(stat, cpu);
+ }
+
+ read_unlock(&tc_lock);
+}
+
+/* Clear all present statistics */
+static int venet_acct_clear_all_stat(void)
+{
+ int ret = -EINTR;
+
+ if (mutex_lock_interruptible(&req_mutex))
+ goto out;
+
+ ret = schedule_on_each_cpu(clear_all_percpu_statistics);
+
+ mutex_unlock(&req_mutex);
+out:
+ return ret;
+}
+
static int venet_acct_get_stat_list(envid_t *__list, int length)
{
int hash;
@@ -748,6 +848,13 @@ static int venet_acct_ioctl(struct file *file, unsigned int cmd,
err = 0;
venet_acct_destroy_all_stat();
break;
+ case VZCTL_TC_CLEAR_STAT:
+ err = venet_acct_clear_stat(arg);
+ break;
+ case VZCTL_TC_CLEAR_ALL_STAT:
+ err = venet_acct_clear_all_stat();
+ break;
+
case VZCTL_TC_GET_BASE:
err = venet_acct_get_base(arg);
break;
diff --git a/kernel/workqueue.c b/kernel/workqueue.c
index 68086a34b8ef..92bc5fb2feac 100644
--- a/kernel/workqueue.c
+++ b/kernel/workqueue.c
@@ -2996,6 +2996,7 @@ int schedule_on_each_cpu(work_func_t func)
free_percpu(works);
return 0;
}
+EXPORT_SYMBOL_GPL(schedule_on_each_cpu);
/**
* flush_scheduled_work - ensure that any scheduled work has run to completion.
--
2.1.4
More information about the Devel
mailing list