/*-*- Mode: C; c-basic-offset: 8; indent-tabs-mode: nil -*-*/

/***
  This file is part of systemd.

  Copyright 2011 Lennart Poettering

  systemd is free software; you can redistribute it and/or modify it
  under the terms of the GNU Lesser General Public License as published by
  the Free Software Foundation; either version 2.1 of the License, or
  (at your option) any later version.

  systemd is distributed in the hope that it will be useful, but
  WITHOUT ANY WARRANTY; without even the implied warranty of
  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  Lesser General Public License for more details.

  You should have received a copy of the GNU Lesser General Public License
  along with systemd; If not, see <http://www.gnu.org/licenses/>.
***/

#include "cgroup-attr.h"
#include "cgroup-util.h"
#include "list.h"
#include "fileio.h"

int cgroup_attribute_apply(CGroupAttribute *a, CGroupBonding *b) {
        _cleanup_free_ char *path = NULL, *v = NULL;
        int r;

        assert(a);

        b = cgroup_bonding_find_list(b, a->controller);
        if (!b)
                return 0;

        if (a->semantics && a->semantics->map_write) {
                r = a->semantics->map_write(a->semantics, a->value, &v);
                if (r < 0)
                        return r;
        }

        r = cg_get_path(a->controller, b->path, a->name, &path);
        if (r < 0)
                return r;

        r = write_string_file(path, v ? v : a->value);
        if (r < 0)
                log_warning("Failed to write '%s' to %s: %s", v ? v : a->value, path, strerror(-r));

        return r;
}

int cgroup_attribute_apply_list(CGroupAttribute *first, CGroupBonding *b) {
        CGroupAttribute *a;
        int r = 0;

        LIST_FOREACH(by_unit, a, first) {
                int k;

                k = cgroup_attribute_apply(a, b);
                if (r == 0)
                        r = k;
        }

        return r;
}

bool cgroup_attribute_matches(CGroupAttribute *a, const char *controller, const char *name) {
        assert(a);

        if (controller) {
                if (streq(a->controller, controller) && (!name || streq(a->name, name)))
                        return true;

        } else if (!name)
                return true;
        else if (streq(a->name, name)) {
                size_t x, y;
                x = strlen(a->controller);
                y = strlen(name);

                if (y > x &&
                    memcmp(a->controller, name, x) == 0 &&
                    name[x] == '.')
                        return true;
        }

        return false;
}

CGroupAttribute *cgroup_attribute_find_list(
                CGroupAttribute *first,
                const char *controller,
                const char *name) {
        CGroupAttribute *a;

        assert(name);

        LIST_FOREACH(by_unit, a, first)
                if (cgroup_attribute_matches(a, controller, name))
                        return a;

        return NULL;
}

void cgroup_attribute_free(CGroupAttribute *a) {
        assert(a);

        if (a->unit)
                LIST_REMOVE(CGroupAttribute, by_unit, a->unit->cgroup_attributes, a);

        free(a->controller);
        free(a->name);
        free(a->value);
        free(a);
}

void cgroup_attribute_free_list(CGroupAttribute *first) {
        CGroupAttribute *a, *n;

        LIST_FOREACH_SAFE(by_unit, a, n, first)
                cgroup_attribute_free(a);
}

void cgroup_attribute_free_some(CGroupAttribute *first, const char *controller, const char *name) {
        CGroupAttribute *a, *n;

        LIST_FOREACH_SAFE(by_unit, a, n, first)
                if (cgroup_attribute_matches(a, controller, name))
                        cgroup_attribute_free(a);
}