1 // SPDX-License-Identifier: GPL-2.0-or-later
3 * AMD MP2 1.1 communication driver
5 * Copyright (c) 2022, Advanced Micro Devices, Inc.
8 * Author: Basavaraj Natikar <Basavaraj.Natikar@amd.com>
11 #include <linux/delay.h>
12 #include <linux/hid.h>
14 #include "amd_sfh_init.h"
15 #include "amd_sfh_interface.h"
16 #include "../hid_descriptor/amd_sfh_hid_desc.h"
18 static int amd_sfh_get_sensor_num(struct amd_mp2_dev
*mp2
, u8
*sensor_id
)
20 struct sfh_sensor_list
*slist
;
21 struct sfh_base_info binfo
;
22 int num_of_sensors
= 0;
25 memcpy_fromio(&binfo
, mp2
->vsbase
, sizeof(struct sfh_base_info
));
26 slist
= &binfo
.sbase
.s_list
;
28 for (i
= 0; i
< MAX_IDX
; i
++) {
35 if (BIT(i
) & slist
->sl
.sensors
)
36 sensor_id
[num_of_sensors
++] = i
;
41 return num_of_sensors
;
44 static u32
amd_sfh_wait_for_response(struct amd_mp2_dev
*mp2
, u8 sid
, u32 cmd_id
)
46 if (mp2
->mp2_ops
->response
)
47 return mp2
->mp2_ops
->response(mp2
, sid
, cmd_id
);
52 static const char *get_sensor_name(int idx
)
56 return "accelerometer";
60 return "magnetometer";
66 return "unknown sensor type";
70 static int amd_sfh_hid_client_deinit(struct amd_mp2_dev
*privdata
)
72 struct amdtp_cl_data
*cl_data
= privdata
->cl_data
;
75 for (i
= 0; i
< cl_data
->num_hid_devices
; i
++) {
76 switch (cl_data
->sensor_idx
[i
]) {
78 privdata
->dev_en
.is_hpd_present
= false;
81 privdata
->dev_en
.is_als_present
= false;
85 if (cl_data
->sensor_sts
[i
] == SENSOR_ENABLED
) {
86 privdata
->mp2_ops
->stop(privdata
, cl_data
->sensor_idx
[i
]);
87 status
= amd_sfh_wait_for_response
88 (privdata
, cl_data
->sensor_idx
[i
], DISABLE_SENSOR
);
90 cl_data
->sensor_sts
[i
] = SENSOR_DISABLED
;
91 dev_dbg(&privdata
->pdev
->dev
, "stopping sid 0x%x (%s) status 0x%x\n",
92 cl_data
->sensor_idx
[i
], get_sensor_name(cl_data
->sensor_idx
[i
]),
93 cl_data
->sensor_sts
[i
]);
97 cancel_delayed_work_sync(&cl_data
->work
);
98 cancel_delayed_work_sync(&cl_data
->work_buffer
);
99 amdtp_hid_remove(cl_data
);
104 static int amd_sfh1_1_hid_client_init(struct amd_mp2_dev
*privdata
)
106 struct amd_input_data
*in_data
= &privdata
->in_data
;
107 struct amdtp_cl_data
*cl_data
= privdata
->cl_data
;
108 struct amd_mp2_ops
*mp2_ops
= privdata
->mp2_ops
;
109 struct amd_mp2_sensor_info info
;
110 struct request_list
*req_list
;
111 u32 feature_report_size
;
112 u32 input_report_size
;
117 req_list
= &cl_data
->req_list
;
118 dev
= &privdata
->pdev
->dev
;
119 amd_sfh1_1_set_desc_ops(mp2_ops
);
121 cl_data
->num_hid_devices
= amd_sfh_get_sensor_num(privdata
, &cl_data
->sensor_idx
[0]);
122 if (cl_data
->num_hid_devices
== 0)
124 cl_data
->is_any_sensor_enabled
= false;
126 INIT_DELAYED_WORK(&cl_data
->work
, amd_sfh_work
);
127 INIT_DELAYED_WORK(&cl_data
->work_buffer
, amd_sfh_work_buffer
);
128 INIT_LIST_HEAD(&req_list
->list
);
129 cl_data
->in_data
= in_data
;
131 for (i
= 0; i
< cl_data
->num_hid_devices
; i
++) {
132 cl_data
->sensor_sts
[i
] = SENSOR_DISABLED
;
133 cl_data
->sensor_requested_cnt
[i
] = 0;
134 cl_data
->cur_hid_dev
= i
;
135 cl_idx
= cl_data
->sensor_idx
[i
];
137 cl_data
->report_descr_sz
[i
] = mp2_ops
->get_desc_sz(cl_idx
, descr_size
);
138 if (!cl_data
->report_descr_sz
[i
]) {
142 feature_report_size
= mp2_ops
->get_desc_sz(cl_idx
, feature_size
);
143 if (!feature_report_size
) {
147 input_report_size
= mp2_ops
->get_desc_sz(cl_idx
, input_size
);
148 if (!input_report_size
) {
152 cl_data
->feature_report
[i
] = devm_kzalloc(dev
, feature_report_size
, GFP_KERNEL
);
153 if (!cl_data
->feature_report
[i
]) {
157 in_data
->input_report
[i
] = devm_kzalloc(dev
, input_report_size
, GFP_KERNEL
);
158 if (!in_data
->input_report
[i
]) {
163 info
.sensor_idx
= cl_idx
;
165 cl_data
->report_descr
[i
] =
166 devm_kzalloc(dev
, cl_data
->report_descr_sz
[i
], GFP_KERNEL
);
167 if (!cl_data
->report_descr
[i
]) {
171 rc
= mp2_ops
->get_rep_desc(cl_idx
, cl_data
->report_descr
[i
]);
175 writel(0, privdata
->mmio
+ amd_get_p2c_val(privdata
, 0));
176 mp2_ops
->start(privdata
, info
);
177 status
= amd_sfh_wait_for_response
178 (privdata
, cl_data
->sensor_idx
[i
], ENABLE_SENSOR
);
180 cl_data
->sensor_sts
[i
] = (status
== 0) ? SENSOR_ENABLED
: SENSOR_DISABLED
;
183 for (i
= 0; i
< cl_data
->num_hid_devices
; i
++) {
184 cl_data
->cur_hid_dev
= i
;
185 if (cl_data
->sensor_sts
[i
] == SENSOR_ENABLED
) {
186 cl_data
->is_any_sensor_enabled
= true;
187 rc
= amdtp_hid_probe(i
, cl_data
);
190 switch (cl_data
->sensor_idx
[i
]) {
192 privdata
->dev_en
.is_hpd_present
= true;
195 privdata
->dev_en
.is_als_present
= true;
199 dev_dbg(dev
, "sid 0x%x (%s) status 0x%x\n",
200 cl_data
->sensor_idx
[i
], get_sensor_name(cl_data
->sensor_idx
[i
]),
201 cl_data
->sensor_sts
[i
]);
204 if (!cl_data
->is_any_sensor_enabled
) {
205 dev_warn(dev
, "No sensor registered, sensors not enabled is %d\n",
206 cl_data
->is_any_sensor_enabled
);
211 schedule_delayed_work(&cl_data
->work_buffer
, msecs_to_jiffies(AMD_SFH_IDLE_LOOP
));
215 amd_sfh_hid_client_deinit(privdata
);
216 for (i
= 0; i
< cl_data
->num_hid_devices
; i
++) {
217 devm_kfree(dev
, cl_data
->feature_report
[i
]);
218 devm_kfree(dev
, in_data
->input_report
[i
]);
219 devm_kfree(dev
, cl_data
->report_descr
[i
]);
224 static void amd_sfh_resume(struct amd_mp2_dev
*mp2
)
226 struct amdtp_cl_data
*cl_data
= mp2
->cl_data
;
227 struct amd_mp2_sensor_info info
;
230 if (!cl_data
->is_any_sensor_enabled
) {
231 amd_sfh_clear_intr(mp2
);
235 for (i
= 0; i
< cl_data
->num_hid_devices
; i
++) {
236 if (cl_data
->sensor_sts
[i
] == SENSOR_DISABLED
) {
237 info
.sensor_idx
= cl_data
->sensor_idx
[i
];
238 mp2
->mp2_ops
->start(mp2
, info
);
239 status
= amd_sfh_wait_for_response
240 (mp2
, cl_data
->sensor_idx
[i
], ENABLE_SENSOR
);
242 status
= SENSOR_ENABLED
;
243 if (status
== SENSOR_ENABLED
)
244 cl_data
->sensor_sts
[i
] = SENSOR_ENABLED
;
245 dev_dbg(&mp2
->pdev
->dev
, "resume sid 0x%x (%s) status 0x%x\n",
246 cl_data
->sensor_idx
[i
], get_sensor_name(cl_data
->sensor_idx
[i
]),
247 cl_data
->sensor_sts
[i
]);
251 schedule_delayed_work(&cl_data
->work_buffer
, msecs_to_jiffies(AMD_SFH_IDLE_LOOP
));
252 amd_sfh_clear_intr(mp2
);
255 static void amd_sfh_suspend(struct amd_mp2_dev
*mp2
)
257 struct amdtp_cl_data
*cl_data
= mp2
->cl_data
;
260 if (!cl_data
->is_any_sensor_enabled
) {
261 amd_sfh_clear_intr(mp2
);
265 for (i
= 0; i
< cl_data
->num_hid_devices
; i
++) {
266 if (cl_data
->sensor_idx
[i
] != HPD_IDX
&&
267 cl_data
->sensor_sts
[i
] == SENSOR_ENABLED
) {
268 mp2
->mp2_ops
->stop(mp2
, cl_data
->sensor_idx
[i
]);
269 status
= amd_sfh_wait_for_response
270 (mp2
, cl_data
->sensor_idx
[i
], DISABLE_SENSOR
);
272 status
= SENSOR_DISABLED
;
273 if (status
!= SENSOR_ENABLED
)
274 cl_data
->sensor_sts
[i
] = SENSOR_DISABLED
;
275 dev_dbg(&mp2
->pdev
->dev
, "suspend sid 0x%x (%s) status 0x%x\n",
276 cl_data
->sensor_idx
[i
], get_sensor_name(cl_data
->sensor_idx
[i
]),
277 cl_data
->sensor_sts
[i
]);
281 cancel_delayed_work_sync(&cl_data
->work_buffer
);
282 amd_sfh_clear_intr(mp2
);
285 static void amd_mp2_pci_remove(void *privdata
)
287 struct amd_mp2_dev
*mp2
= privdata
;
290 amd_sfh_hid_client_deinit(privdata
);
291 mp2
->mp2_ops
->stop_all(mp2
);
292 pci_intx(mp2
->pdev
, false);
293 amd_sfh_clear_intr(mp2
);
296 static void amd_sfh_set_ops(struct amd_mp2_dev
*mp2
)
298 struct amd_mp2_ops
*mp2_ops
;
300 sfh_interface_init(mp2
);
301 mp2_ops
= mp2
->mp2_ops
;
302 mp2_ops
->clear_intr
= amd_sfh_clear_intr_v2
;
303 mp2_ops
->init_intr
= amd_sfh_irq_init_v2
;
304 mp2_ops
->suspend
= amd_sfh_suspend
;
305 mp2_ops
->resume
= amd_sfh_resume
;
306 mp2_ops
->remove
= amd_mp2_pci_remove
;
309 int amd_sfh1_1_init(struct amd_mp2_dev
*mp2
)
311 u32 phy_base
= readl(mp2
->mmio
+ amd_get_c2p_val(mp2
, 22));
312 struct device
*dev
= &mp2
->pdev
->dev
;
313 struct sfh_base_info binfo
;
317 if (!devm_request_mem_region(dev
, phy_base
, 128 * 1024, "amd_sfh")) {
318 dev_dbg(dev
, "can't reserve mmio registers\n");
322 mp2
->vsbase
= devm_ioremap(dev
, phy_base
, 128 * 1024);
324 dev_dbg(dev
, "failed to remap vsbase\n");
328 /* Before accessing give time for SFH firmware for processing configuration */
331 memcpy_fromio(&binfo
, mp2
->vsbase
, sizeof(struct sfh_base_info
));
332 if (binfo
.sbase
.fw_info
.fw_ver
== 0 || binfo
.sbase
.s_list
.sl
.sensors
== 0) {
333 dev_dbg(dev
, "No sensor registered\n");
336 dev_dbg(dev
, "firmware version 0x%x\n", binfo
.sbase
.fw_info
.fw_ver
);
338 amd_sfh_set_ops(mp2
);
340 rc
= amd_sfh_irq_init(mp2
);
343 dev_err(dev
, "amd_sfh_irq_init failed\n");
347 rc
= amd_sfh1_1_hid_client_init(mp2
);
350 if ((rc
!= -ENODEV
) && (rc
!= -EOPNOTSUPP
))
351 dev_err(dev
, "amd_sfh1_1_hid_client_init failed\n");