1 // SPDX-License-Identifier: GPL-2.0+
2 /*
3  * Surface Platform Profile / Performance Mode driver for Surface System
4  * Aggregator Module (thermal and fan subsystem).
5  *
6  * Copyright (C) 2021-2022 Maximilian Luz <luzmaximilian@gmail.com>
7  */
8 
9 #include <linux/unaligned.h>
10 #include <linux/kernel.h>
11 #include <linux/module.h>
12 #include <linux/platform_profile.h>
13 #include <linux/types.h>
14 
15 #include <linux/surface_aggregator/device.h>
16 
17 // Enum for the platform performance profile sent to the TMP module.
18 enum ssam_tmp_profile {
19 	SSAM_TMP_PROFILE_NORMAL             = 1,
20 	SSAM_TMP_PROFILE_BATTERY_SAVER      = 2,
21 	SSAM_TMP_PROFILE_BETTER_PERFORMANCE = 3,
22 	SSAM_TMP_PROFILE_BEST_PERFORMANCE   = 4,
23 };
24 
25 // Enum for the fan profile sent to the FAN module. This fan profile is
26 // only sent to the EC if the 'has_fan' property is set. The integers are
27 // not a typo, they differ from the performance profile indices.
28 enum ssam_fan_profile {
29 	SSAM_FAN_PROFILE_NORMAL             = 2,
30 	SSAM_FAN_PROFILE_BATTERY_SAVER      = 1,
31 	SSAM_FAN_PROFILE_BETTER_PERFORMANCE = 3,
32 	SSAM_FAN_PROFILE_BEST_PERFORMANCE   = 4,
33 };
34 
35 struct ssam_tmp_profile_info {
36 	__le32 profile;
37 	__le16 unknown1;
38 	__le16 unknown2;
39 } __packed;
40 
41 struct ssam_platform_profile_device {
42 	struct ssam_device *sdev;
43 	struct platform_profile_handler handler;
44 	bool has_fan;
45 };
46 
47 SSAM_DEFINE_SYNC_REQUEST_CL_R(__ssam_tmp_profile_get, struct ssam_tmp_profile_info, {
48 	.target_category = SSAM_SSH_TC_TMP,
49 	.command_id      = 0x02,
50 });
51 
52 SSAM_DEFINE_SYNC_REQUEST_CL_W(__ssam_tmp_profile_set, __le32, {
53 	.target_category = SSAM_SSH_TC_TMP,
54 	.command_id      = 0x03,
55 });
56 
57 SSAM_DEFINE_SYNC_REQUEST_W(__ssam_fan_profile_set, u8, {
58 	.target_category = SSAM_SSH_TC_FAN,
59 	.target_id = SSAM_SSH_TID_SAM,
60 	.command_id = 0x0e,
61 	.instance_id = 0x01,
62 });
63 
ssam_tmp_profile_get(struct ssam_device * sdev,enum ssam_tmp_profile * p)64 static int ssam_tmp_profile_get(struct ssam_device *sdev, enum ssam_tmp_profile *p)
65 {
66 	struct ssam_tmp_profile_info info;
67 	int status;
68 
69 	status = ssam_retry(__ssam_tmp_profile_get, sdev, &info);
70 	if (status < 0)
71 		return status;
72 
73 	*p = le32_to_cpu(info.profile);
74 	return 0;
75 }
76 
ssam_tmp_profile_set(struct ssam_device * sdev,enum ssam_tmp_profile p)77 static int ssam_tmp_profile_set(struct ssam_device *sdev, enum ssam_tmp_profile p)
78 {
79 	const __le32 profile_le = cpu_to_le32(p);
80 
81 	return ssam_retry(__ssam_tmp_profile_set, sdev, &profile_le);
82 }
83 
ssam_fan_profile_set(struct ssam_device * sdev,enum ssam_fan_profile p)84 static int ssam_fan_profile_set(struct ssam_device *sdev, enum ssam_fan_profile p)
85 {
86 	const u8 profile = p;
87 
88 	return ssam_retry(__ssam_fan_profile_set, sdev->ctrl, &profile);
89 }
90 
convert_ssam_tmp_to_profile(struct ssam_device * sdev,enum ssam_tmp_profile p)91 static int convert_ssam_tmp_to_profile(struct ssam_device *sdev, enum ssam_tmp_profile p)
92 {
93 	switch (p) {
94 	case SSAM_TMP_PROFILE_NORMAL:
95 		return PLATFORM_PROFILE_BALANCED;
96 
97 	case SSAM_TMP_PROFILE_BATTERY_SAVER:
98 		return PLATFORM_PROFILE_LOW_POWER;
99 
100 	case SSAM_TMP_PROFILE_BETTER_PERFORMANCE:
101 		return PLATFORM_PROFILE_BALANCED_PERFORMANCE;
102 
103 	case SSAM_TMP_PROFILE_BEST_PERFORMANCE:
104 		return PLATFORM_PROFILE_PERFORMANCE;
105 
106 	default:
107 		dev_err(&sdev->dev, "invalid performance profile: %d", p);
108 		return -EINVAL;
109 	}
110 }
111 
112 
convert_profile_to_ssam_tmp(struct ssam_device * sdev,enum platform_profile_option p)113 static int convert_profile_to_ssam_tmp(struct ssam_device *sdev, enum platform_profile_option p)
114 {
115 	switch (p) {
116 	case PLATFORM_PROFILE_LOW_POWER:
117 		return SSAM_TMP_PROFILE_BATTERY_SAVER;
118 
119 	case PLATFORM_PROFILE_BALANCED:
120 		return SSAM_TMP_PROFILE_NORMAL;
121 
122 	case PLATFORM_PROFILE_BALANCED_PERFORMANCE:
123 		return SSAM_TMP_PROFILE_BETTER_PERFORMANCE;
124 
125 	case PLATFORM_PROFILE_PERFORMANCE:
126 		return SSAM_TMP_PROFILE_BEST_PERFORMANCE;
127 
128 	default:
129 		/* This should have already been caught by platform_profile_store(). */
130 		WARN(true, "unsupported platform profile");
131 		return -EOPNOTSUPP;
132 	}
133 }
134 
convert_profile_to_ssam_fan(struct ssam_device * sdev,enum platform_profile_option p)135 static int convert_profile_to_ssam_fan(struct ssam_device *sdev, enum platform_profile_option p)
136 {
137 	switch (p) {
138 	case PLATFORM_PROFILE_LOW_POWER:
139 		return SSAM_FAN_PROFILE_BATTERY_SAVER;
140 
141 	case PLATFORM_PROFILE_BALANCED:
142 		return SSAM_FAN_PROFILE_NORMAL;
143 
144 	case PLATFORM_PROFILE_BALANCED_PERFORMANCE:
145 		return SSAM_FAN_PROFILE_BETTER_PERFORMANCE;
146 
147 	case PLATFORM_PROFILE_PERFORMANCE:
148 		return SSAM_FAN_PROFILE_BEST_PERFORMANCE;
149 
150 	default:
151 		/* This should have already been caught by platform_profile_store(). */
152 		WARN(true, "unsupported platform profile");
153 		return -EOPNOTSUPP;
154 	}
155 }
156 
ssam_platform_profile_get(struct platform_profile_handler * pprof,enum platform_profile_option * profile)157 static int ssam_platform_profile_get(struct platform_profile_handler *pprof,
158 				     enum platform_profile_option *profile)
159 {
160 	struct ssam_platform_profile_device *tpd;
161 	enum ssam_tmp_profile tp;
162 	int status;
163 
164 	tpd = container_of(pprof, struct ssam_platform_profile_device, handler);
165 
166 	status = ssam_tmp_profile_get(tpd->sdev, &tp);
167 	if (status)
168 		return status;
169 
170 	status = convert_ssam_tmp_to_profile(tpd->sdev, tp);
171 	if (status < 0)
172 		return status;
173 
174 	*profile = status;
175 	return 0;
176 }
177 
ssam_platform_profile_set(struct platform_profile_handler * pprof,enum platform_profile_option profile)178 static int ssam_platform_profile_set(struct platform_profile_handler *pprof,
179 				     enum platform_profile_option profile)
180 {
181 	struct ssam_platform_profile_device *tpd;
182 	int tp;
183 
184 	tpd = container_of(pprof, struct ssam_platform_profile_device, handler);
185 
186 	tp = convert_profile_to_ssam_tmp(tpd->sdev, profile);
187 	if (tp < 0)
188 		return tp;
189 
190 	tp = ssam_tmp_profile_set(tpd->sdev, tp);
191 	if (tp < 0)
192 		return tp;
193 
194 	if (tpd->has_fan) {
195 		tp = convert_profile_to_ssam_fan(tpd->sdev, profile);
196 		if (tp < 0)
197 			return tp;
198 		tp = ssam_fan_profile_set(tpd->sdev, tp);
199 	}
200 
201 	return tp;
202 }
203 
surface_platform_profile_probe(struct ssam_device * sdev)204 static int surface_platform_profile_probe(struct ssam_device *sdev)
205 {
206 	struct ssam_platform_profile_device *tpd;
207 
208 	tpd = devm_kzalloc(&sdev->dev, sizeof(*tpd), GFP_KERNEL);
209 	if (!tpd)
210 		return -ENOMEM;
211 
212 	tpd->sdev = sdev;
213 
214 	tpd->handler.profile_get = ssam_platform_profile_get;
215 	tpd->handler.profile_set = ssam_platform_profile_set;
216 
217 	tpd->has_fan = device_property_read_bool(&sdev->dev, "has_fan");
218 
219 	set_bit(PLATFORM_PROFILE_LOW_POWER, tpd->handler.choices);
220 	set_bit(PLATFORM_PROFILE_BALANCED, tpd->handler.choices);
221 	set_bit(PLATFORM_PROFILE_BALANCED_PERFORMANCE, tpd->handler.choices);
222 	set_bit(PLATFORM_PROFILE_PERFORMANCE, tpd->handler.choices);
223 
224 	return platform_profile_register(&tpd->handler);
225 }
226 
surface_platform_profile_remove(struct ssam_device * sdev)227 static void surface_platform_profile_remove(struct ssam_device *sdev)
228 {
229 	platform_profile_remove();
230 }
231 
232 static const struct ssam_device_id ssam_platform_profile_match[] = {
233 	{ SSAM_SDEV(TMP, SAM, 0x00, 0x01) },
234 	{ },
235 };
236 MODULE_DEVICE_TABLE(ssam, ssam_platform_profile_match);
237 
238 static struct ssam_device_driver surface_platform_profile = {
239 	.probe = surface_platform_profile_probe,
240 	.remove = surface_platform_profile_remove,
241 	.match_table = ssam_platform_profile_match,
242 	.driver = {
243 		.name = "surface_platform_profile",
244 		.probe_type = PROBE_PREFER_ASYNCHRONOUS,
245 	},
246 };
247 module_ssam_device_driver(surface_platform_profile);
248 
249 MODULE_AUTHOR("Maximilian Luz <luzmaximilian@gmail.com>");
250 MODULE_DESCRIPTION("Platform Profile Support for Surface System Aggregator Module");
251 MODULE_LICENSE("GPL");
252