xref: /openbmc/linux/drivers/acpi/fan_attr.c (revision 7ae9fb1b7ecbb5d85d07857943f677fd1a559b18)
100ae053aSSrinivas Pandruvada // SPDX-License-Identifier: GPL-2.0-or-later
200ae053aSSrinivas Pandruvada /*
300ae053aSSrinivas Pandruvada  *  fan_attr.c - Create extra attributes for ACPI Fan driver
400ae053aSSrinivas Pandruvada  *
500ae053aSSrinivas Pandruvada  *  Copyright (C) 2001, 2002 Andy Grover <andrew.grover@intel.com>
600ae053aSSrinivas Pandruvada  *  Copyright (C) 2001, 2002 Paul Diefenbaugh <paul.s.diefenbaugh@intel.com>
700ae053aSSrinivas Pandruvada  *  Copyright (C) 2022 Intel Corporation. All rights reserved.
800ae053aSSrinivas Pandruvada  */
900ae053aSSrinivas Pandruvada 
1000ae053aSSrinivas Pandruvada #include <linux/kernel.h>
1100ae053aSSrinivas Pandruvada #include <linux/module.h>
1200ae053aSSrinivas Pandruvada #include <linux/init.h>
1300ae053aSSrinivas Pandruvada #include <linux/acpi.h>
1400ae053aSSrinivas Pandruvada 
1500ae053aSSrinivas Pandruvada #include "fan.h"
1600ae053aSSrinivas Pandruvada 
1700ae053aSSrinivas Pandruvada MODULE_LICENSE("GPL");
1800ae053aSSrinivas Pandruvada 
show_state(struct device * dev,struct device_attribute * attr,char * buf)1900ae053aSSrinivas Pandruvada static ssize_t show_state(struct device *dev, struct device_attribute *attr, char *buf)
2000ae053aSSrinivas Pandruvada {
2100ae053aSSrinivas Pandruvada 	struct acpi_fan_fps *fps = container_of(attr, struct acpi_fan_fps, dev_attr);
2200ae053aSSrinivas Pandruvada 	int count;
2300ae053aSSrinivas Pandruvada 
2400ae053aSSrinivas Pandruvada 	if (fps->control == 0xFFFFFFFF || fps->control > 100)
2500ae053aSSrinivas Pandruvada 		count = scnprintf(buf, PAGE_SIZE, "not-defined:");
2600ae053aSSrinivas Pandruvada 	else
2700ae053aSSrinivas Pandruvada 		count = scnprintf(buf, PAGE_SIZE, "%lld:", fps->control);
2800ae053aSSrinivas Pandruvada 
2900ae053aSSrinivas Pandruvada 	if (fps->trip_point == 0xFFFFFFFF || fps->trip_point > 9)
30*64ee2528Sye xingchen 		count += sysfs_emit_at(buf, count, "not-defined:");
3100ae053aSSrinivas Pandruvada 	else
32*64ee2528Sye xingchen 		count += sysfs_emit_at(buf, count, "%lld:", fps->trip_point);
3300ae053aSSrinivas Pandruvada 
3400ae053aSSrinivas Pandruvada 	if (fps->speed == 0xFFFFFFFF)
35*64ee2528Sye xingchen 		count += sysfs_emit_at(buf, count, "not-defined:");
3600ae053aSSrinivas Pandruvada 	else
37*64ee2528Sye xingchen 		count += sysfs_emit_at(buf, count, "%lld:", fps->speed);
3800ae053aSSrinivas Pandruvada 
3900ae053aSSrinivas Pandruvada 	if (fps->noise_level == 0xFFFFFFFF)
40*64ee2528Sye xingchen 		count += sysfs_emit_at(buf, count, "not-defined:");
4100ae053aSSrinivas Pandruvada 	else
42*64ee2528Sye xingchen 		count += sysfs_emit_at(buf, count, "%lld:", fps->noise_level * 100);
4300ae053aSSrinivas Pandruvada 
4400ae053aSSrinivas Pandruvada 	if (fps->power == 0xFFFFFFFF)
45*64ee2528Sye xingchen 		count += sysfs_emit_at(buf, count, "not-defined\n");
4600ae053aSSrinivas Pandruvada 	else
47*64ee2528Sye xingchen 		count += sysfs_emit_at(buf, count, "%lld\n", fps->power);
4800ae053aSSrinivas Pandruvada 
4900ae053aSSrinivas Pandruvada 	return count;
5000ae053aSSrinivas Pandruvada }
5100ae053aSSrinivas Pandruvada 
show_fan_speed(struct device * dev,struct device_attribute * attr,char * buf)52f1197343SSrinivas Pandruvada static ssize_t show_fan_speed(struct device *dev, struct device_attribute *attr, char *buf)
53f1197343SSrinivas Pandruvada {
54f1197343SSrinivas Pandruvada 	struct acpi_device *acpi_dev = container_of(dev, struct acpi_device, dev);
55f1197343SSrinivas Pandruvada 	struct acpi_fan_fst fst;
56f1197343SSrinivas Pandruvada 	int status;
57f1197343SSrinivas Pandruvada 
58f1197343SSrinivas Pandruvada 	status = acpi_fan_get_fst(acpi_dev, &fst);
59f1197343SSrinivas Pandruvada 	if (status)
60f1197343SSrinivas Pandruvada 		return status;
61f1197343SSrinivas Pandruvada 
62f1197343SSrinivas Pandruvada 	return sprintf(buf, "%lld\n", fst.speed);
63f1197343SSrinivas Pandruvada }
64f1197343SSrinivas Pandruvada 
show_fine_grain_control(struct device * dev,struct device_attribute * attr,char * buf)65f1197343SSrinivas Pandruvada static ssize_t show_fine_grain_control(struct device *dev, struct device_attribute *attr, char *buf)
66f1197343SSrinivas Pandruvada {
67f1197343SSrinivas Pandruvada 	struct acpi_device *acpi_dev = container_of(dev, struct acpi_device, dev);
68f1197343SSrinivas Pandruvada 	struct acpi_fan *fan = acpi_driver_data(acpi_dev);
69f1197343SSrinivas Pandruvada 
70f1197343SSrinivas Pandruvada 	return sprintf(buf, "%d\n", fan->fif.fine_grain_ctrl);
71f1197343SSrinivas Pandruvada }
72f1197343SSrinivas Pandruvada 
acpi_fan_create_attributes(struct acpi_device * device)7300ae053aSSrinivas Pandruvada int acpi_fan_create_attributes(struct acpi_device *device)
7400ae053aSSrinivas Pandruvada {
7500ae053aSSrinivas Pandruvada 	struct acpi_fan *fan = acpi_driver_data(device);
76f1197343SSrinivas Pandruvada 	int i, status;
77f1197343SSrinivas Pandruvada 
78f1197343SSrinivas Pandruvada 	sysfs_attr_init(&fan->fine_grain_control.attr);
79f1197343SSrinivas Pandruvada 	fan->fine_grain_control.show = show_fine_grain_control;
80f1197343SSrinivas Pandruvada 	fan->fine_grain_control.store = NULL;
81f1197343SSrinivas Pandruvada 	fan->fine_grain_control.attr.name = "fine_grain_control";
82f1197343SSrinivas Pandruvada 	fan->fine_grain_control.attr.mode = 0444;
83f1197343SSrinivas Pandruvada 	status = sysfs_create_file(&device->dev.kobj, &fan->fine_grain_control.attr);
84f1197343SSrinivas Pandruvada 	if (status)
85f1197343SSrinivas Pandruvada 		return status;
86f1197343SSrinivas Pandruvada 
87f1197343SSrinivas Pandruvada 	/* _FST is present if we are here */
88f1197343SSrinivas Pandruvada 	sysfs_attr_init(&fan->fst_speed.attr);
89f1197343SSrinivas Pandruvada 	fan->fst_speed.show = show_fan_speed;
90f1197343SSrinivas Pandruvada 	fan->fst_speed.store = NULL;
91f1197343SSrinivas Pandruvada 	fan->fst_speed.attr.name = "fan_speed_rpm";
92f1197343SSrinivas Pandruvada 	fan->fst_speed.attr.mode = 0444;
93f1197343SSrinivas Pandruvada 	status = sysfs_create_file(&device->dev.kobj, &fan->fst_speed.attr);
94f1197343SSrinivas Pandruvada 	if (status)
95f1197343SSrinivas Pandruvada 		goto rem_fine_grain_attr;
9600ae053aSSrinivas Pandruvada 
9700ae053aSSrinivas Pandruvada 	for (i = 0; i < fan->fps_count; ++i) {
9800ae053aSSrinivas Pandruvada 		struct acpi_fan_fps *fps = &fan->fps[i];
9900ae053aSSrinivas Pandruvada 
10000ae053aSSrinivas Pandruvada 		snprintf(fps->name, ACPI_FPS_NAME_LEN, "state%d", i);
10100ae053aSSrinivas Pandruvada 		sysfs_attr_init(&fps->dev_attr.attr);
10200ae053aSSrinivas Pandruvada 		fps->dev_attr.show = show_state;
10300ae053aSSrinivas Pandruvada 		fps->dev_attr.store = NULL;
10400ae053aSSrinivas Pandruvada 		fps->dev_attr.attr.name = fps->name;
10500ae053aSSrinivas Pandruvada 		fps->dev_attr.attr.mode = 0444;
10600ae053aSSrinivas Pandruvada 		status = sysfs_create_file(&device->dev.kobj, &fps->dev_attr.attr);
10700ae053aSSrinivas Pandruvada 		if (status) {
10800ae053aSSrinivas Pandruvada 			int j;
10900ae053aSSrinivas Pandruvada 
11000ae053aSSrinivas Pandruvada 			for (j = 0; j < i; ++j)
11100ae053aSSrinivas Pandruvada 				sysfs_remove_file(&device->dev.kobj, &fan->fps[j].dev_attr.attr);
112f1197343SSrinivas Pandruvada 			goto rem_fst_attr;
11300ae053aSSrinivas Pandruvada 		}
11400ae053aSSrinivas Pandruvada 	}
11500ae053aSSrinivas Pandruvada 
116f1197343SSrinivas Pandruvada 	return 0;
117f1197343SSrinivas Pandruvada 
118f1197343SSrinivas Pandruvada rem_fst_attr:
119f1197343SSrinivas Pandruvada 	sysfs_remove_file(&device->dev.kobj, &fan->fst_speed.attr);
120f1197343SSrinivas Pandruvada 
121f1197343SSrinivas Pandruvada rem_fine_grain_attr:
122f1197343SSrinivas Pandruvada 	sysfs_remove_file(&device->dev.kobj, &fan->fine_grain_control.attr);
123f1197343SSrinivas Pandruvada 
12400ae053aSSrinivas Pandruvada 	return status;
12500ae053aSSrinivas Pandruvada }
12600ae053aSSrinivas Pandruvada 
acpi_fan_delete_attributes(struct acpi_device * device)12700ae053aSSrinivas Pandruvada void acpi_fan_delete_attributes(struct acpi_device *device)
12800ae053aSSrinivas Pandruvada {
12900ae053aSSrinivas Pandruvada 	struct acpi_fan *fan = acpi_driver_data(device);
13000ae053aSSrinivas Pandruvada 	int i;
13100ae053aSSrinivas Pandruvada 
13200ae053aSSrinivas Pandruvada 	for (i = 0; i < fan->fps_count; ++i)
13300ae053aSSrinivas Pandruvada 		sysfs_remove_file(&device->dev.kobj, &fan->fps[i].dev_attr.attr);
134f1197343SSrinivas Pandruvada 
135f1197343SSrinivas Pandruvada 	sysfs_remove_file(&device->dev.kobj, &fan->fst_speed.attr);
136f1197343SSrinivas Pandruvada 	sysfs_remove_file(&device->dev.kobj, &fan->fine_grain_control.attr);
13700ae053aSSrinivas Pandruvada }
138