| /* |
| * Abstract code for CPUFreq governor tunable sysfs attributes. |
| * |
| * Copyright (C) 2016, Intel Corporation |
| * Author: Rafael J. Wysocki <rafael.j.wysocki@intel.com> |
| * |
| * This program is free software; you can redistribute it and/or modify |
| * it under the terms of the GNU General Public License version 2 as |
| * published by the Free Software Foundation. |
| */ |
| |
| #include "cpufreq_governor.h" |
| |
| static inline struct gov_attr_set *to_gov_attr_set(struct kobject *kobj) |
| { |
| return container_of(kobj, struct gov_attr_set, kobj); |
| } |
| |
| static inline struct governor_attr *to_gov_attr(struct attribute *attr) |
| { |
| return container_of(attr, struct governor_attr, attr); |
| } |
| |
| static ssize_t governor_show(struct kobject *kobj, struct attribute *attr, |
| char *buf) |
| { |
| struct governor_attr *gattr = to_gov_attr(attr); |
| |
| return gattr->show(to_gov_attr_set(kobj), buf); |
| } |
| |
| static ssize_t governor_store(struct kobject *kobj, struct attribute *attr, |
| const char *buf, size_t count) |
| { |
| struct gov_attr_set *attr_set = to_gov_attr_set(kobj); |
| struct governor_attr *gattr = to_gov_attr(attr); |
| int ret; |
| |
| mutex_lock(&attr_set->update_lock); |
| ret = attr_set->usage_count ? gattr->store(attr_set, buf, count) : -EBUSY; |
| mutex_unlock(&attr_set->update_lock); |
| return ret; |
| } |
| |
| const struct sysfs_ops governor_sysfs_ops = { |
| .show = governor_show, |
| .store = governor_store, |
| }; |
| EXPORT_SYMBOL_GPL(governor_sysfs_ops); |
| |
| void gov_attr_set_init(struct gov_attr_set *attr_set, struct list_head *list_node) |
| { |
| INIT_LIST_HEAD(&attr_set->policy_list); |
| mutex_init(&attr_set->update_lock); |
| attr_set->usage_count = 1; |
| list_add(list_node, &attr_set->policy_list); |
| } |
| EXPORT_SYMBOL_GPL(gov_attr_set_init); |
| |
| void gov_attr_set_get(struct gov_attr_set *attr_set, struct list_head *list_node) |
| { |
| mutex_lock(&attr_set->update_lock); |
| attr_set->usage_count++; |
| list_add(list_node, &attr_set->policy_list); |
| mutex_unlock(&attr_set->update_lock); |
| } |
| EXPORT_SYMBOL_GPL(gov_attr_set_get); |
| |
| unsigned int gov_attr_set_put(struct gov_attr_set *attr_set, struct list_head *list_node) |
| { |
| unsigned int count; |
| |
| mutex_lock(&attr_set->update_lock); |
| list_del(list_node); |
| count = --attr_set->usage_count; |
| mutex_unlock(&attr_set->update_lock); |
| if (count) |
| return count; |
| |
| kobject_put(&attr_set->kobj); |
| mutex_destroy(&attr_set->update_lock); |
| return 0; |
| } |
| EXPORT_SYMBOL_GPL(gov_attr_set_put); |