[patch 6/8] res_counter: add interface for maximum nofail charge

From: David Rientjes
Date: Wed Dec 04 2013 - 00:21:15 EST


For memcg oom reserves, we'll need a resource counter interface that will
not fail when exceeding the memcg limit like res_counter_charge_nofail,
but only to a ceiling.

This patch adds res_counter_charge_nofail_max() that will exceed the
resource counter but only to a maximum defined value. If it fails to
charge the resource, it returns -ENOMEM.

Signed-off-by: David Rientjes <rientjes@xxxxxxxxxx>
---
include/linux/res_counter.h | 10 +++++++++-
kernel/res_counter.c | 27 +++++++++++++++++++++------
2 files changed, 30 insertions(+), 7 deletions(-)

diff --git a/include/linux/res_counter.h b/include/linux/res_counter.h
--- a/include/linux/res_counter.h
+++ b/include/linux/res_counter.h
@@ -107,14 +107,22 @@ void res_counter_init(struct res_counter *counter, struct res_counter *parent);
* counter->limit
*
* charge_nofail works the same, except that it charges the resource
- * counter unconditionally, and returns < 0 if the after the current
+ * counter unconditionally, and returns < 0 if after the current
* charge we are over limit.
+ *
+ * charge_nofail_max is the same as charge_nofail, except that the
+ * resource counter usage can only exceed the limit by the max
+ * difference. Unlike charge_nofail, charge_nofail_max returns < 0
+ * only if the current charge fails because of the max difference.
*/

int __must_check res_counter_charge(struct res_counter *counter,
unsigned long val, struct res_counter **limit_fail_at);
int res_counter_charge_nofail(struct res_counter *counter,
unsigned long val, struct res_counter **limit_fail_at);
+int res_counter_charge_nofail_max(struct res_counter *counter,
+ unsigned long val, struct res_counter **limit_fail_at,
+ unsigned long max);

/*
* uncharge - tell that some portion of the resource is released
diff --git a/kernel/res_counter.c b/kernel/res_counter.c
--- a/kernel/res_counter.c
+++ b/kernel/res_counter.c
@@ -33,15 +33,19 @@ static u64 res_counter_uncharge_locked(struct res_counter *counter,
}

static int res_counter_charge_locked(struct res_counter *counter,
- unsigned long val, bool force)
+ unsigned long val, bool force,
+ unsigned long max)
{
int ret = 0;

if (counter->usage + val > counter->limit) {
counter->failcnt++;
- ret = -ENOMEM;
+ if (max == ULONG_MAX)
+ ret = -ENOMEM;
if (!force)
return ret;
+ if (counter->usage + val - counter->limit > max)
+ return -ENOMEM;
}

counter->usage += val;
@@ -51,7 +55,8 @@ static int res_counter_charge_locked(struct res_counter *counter,
}

static int __res_counter_charge(struct res_counter *counter, unsigned long val,
- struct res_counter **limit_fail_at, bool force)
+ struct res_counter **limit_fail_at, bool force,
+ unsigned long max)
{
int ret, r;
unsigned long flags;
@@ -62,7 +67,7 @@ static int __res_counter_charge(struct res_counter *counter, unsigned long val,
local_irq_save(flags);
for (c = counter; c != NULL; c = c->parent) {
spin_lock(&c->lock);
- r = res_counter_charge_locked(c, val, force);
+ r = res_counter_charge_locked(c, val, force, max);
spin_unlock(&c->lock);
if (r < 0 && !ret) {
ret = r;
@@ -87,13 +92,23 @@ static int __res_counter_charge(struct res_counter *counter, unsigned long val,
int res_counter_charge(struct res_counter *counter, unsigned long val,
struct res_counter **limit_fail_at)
{
- return __res_counter_charge(counter, val, limit_fail_at, false);
+ return __res_counter_charge(counter, val, limit_fail_at, false,
+ ULONG_MAX);
}

int res_counter_charge_nofail(struct res_counter *counter, unsigned long val,
struct res_counter **limit_fail_at)
{
- return __res_counter_charge(counter, val, limit_fail_at, true);
+ return __res_counter_charge(counter, val, limit_fail_at, true,
+ ULONG_MAX);
+}
+
+int res_counter_charge_nofail_max(struct res_counter *counter,
+ unsigned long val,
+ struct res_counter **limit_fail_at,
+ unsigned long max)
+{
+ return __res_counter_charge(counter, val, limit_fail_at, true, max);
}

u64 res_counter_uncharge_until(struct res_counter *counter,
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/