xref: /openbmc/linux/drivers/crypto/intel/qat/qat_common/adf_heartbeat.c (revision c900529f3d9161bfde5cca0754f83b4d3c3e0220)
1359b84f8SDamian Muszynski // SPDX-License-Identifier: GPL-2.0-only
2359b84f8SDamian Muszynski /* Copyright(c) 2023 Intel Corporation */
3359b84f8SDamian Muszynski 
4359b84f8SDamian Muszynski #include <linux/dev_printk.h>
5359b84f8SDamian Muszynski #include <linux/dma-mapping.h>
6359b84f8SDamian Muszynski #include <linux/export.h>
7359b84f8SDamian Muszynski #include <linux/kernel.h>
8359b84f8SDamian Muszynski #include <linux/kstrtox.h>
9359b84f8SDamian Muszynski #include <linux/overflow.h>
10359b84f8SDamian Muszynski #include <linux/string.h>
11359b84f8SDamian Muszynski #include <linux/slab.h>
12359b84f8SDamian Muszynski #include <linux/types.h>
13359b84f8SDamian Muszynski #include <asm/errno.h>
14359b84f8SDamian Muszynski #include "adf_accel_devices.h"
15359b84f8SDamian Muszynski #include "adf_cfg.h"
16359b84f8SDamian Muszynski #include "adf_cfg_strings.h"
17359b84f8SDamian Muszynski #include "adf_clock.h"
18359b84f8SDamian Muszynski #include "adf_common_drv.h"
19359b84f8SDamian Muszynski #include "adf_heartbeat.h"
20359b84f8SDamian Muszynski #include "adf_transport_internal.h"
21359b84f8SDamian Muszynski #include "icp_qat_fw_init_admin.h"
22359b84f8SDamian Muszynski 
23*bec61a29SDamian Muszynski #define ADF_HB_EMPTY_SIG 0xA5A5A5A5
24*bec61a29SDamian Muszynski 
25359b84f8SDamian Muszynski /* Heartbeat counter pair */
26359b84f8SDamian Muszynski struct hb_cnt_pair {
27359b84f8SDamian Muszynski 	__u16 resp_heartbeat_cnt;
28359b84f8SDamian Muszynski 	__u16 req_heartbeat_cnt;
29359b84f8SDamian Muszynski };
30359b84f8SDamian Muszynski 
adf_hb_check_polling_freq(struct adf_accel_dev * accel_dev)31359b84f8SDamian Muszynski static int adf_hb_check_polling_freq(struct adf_accel_dev *accel_dev)
32359b84f8SDamian Muszynski {
33359b84f8SDamian Muszynski 	u64 curr_time = adf_clock_get_current_time();
34359b84f8SDamian Muszynski 	u64 polling_time = curr_time - accel_dev->heartbeat->last_hb_check_time;
35359b84f8SDamian Muszynski 
36359b84f8SDamian Muszynski 	if (polling_time < accel_dev->heartbeat->hb_timer) {
37359b84f8SDamian Muszynski 		dev_warn(&GET_DEV(accel_dev),
38359b84f8SDamian Muszynski 			 "HB polling too frequent. Configured HB timer %d ms\n",
39359b84f8SDamian Muszynski 			 accel_dev->heartbeat->hb_timer);
40359b84f8SDamian Muszynski 		return -EINVAL;
41359b84f8SDamian Muszynski 	}
42359b84f8SDamian Muszynski 
43359b84f8SDamian Muszynski 	accel_dev->heartbeat->last_hb_check_time = curr_time;
44359b84f8SDamian Muszynski 	return 0;
45359b84f8SDamian Muszynski }
46359b84f8SDamian Muszynski 
47*bec61a29SDamian Muszynski /**
48*bec61a29SDamian Muszynski  * validate_hb_ctrs_cnt() - checks if the number of heartbeat counters should
49*bec61a29SDamian Muszynski  * be updated by one to support the currently loaded firmware.
50*bec61a29SDamian Muszynski  * @accel_dev: Pointer to acceleration device.
51*bec61a29SDamian Muszynski  *
52*bec61a29SDamian Muszynski  * Return:
53*bec61a29SDamian Muszynski  * * true - hb_ctrs must increased by ADF_NUM_PKE_STRAND
54*bec61a29SDamian Muszynski  * * false - no changes needed
55*bec61a29SDamian Muszynski  */
validate_hb_ctrs_cnt(struct adf_accel_dev * accel_dev)56*bec61a29SDamian Muszynski static bool validate_hb_ctrs_cnt(struct adf_accel_dev *accel_dev)
57*bec61a29SDamian Muszynski {
58*bec61a29SDamian Muszynski 	const size_t hb_ctrs = accel_dev->hw_device->num_hb_ctrs;
59*bec61a29SDamian Muszynski 	const size_t max_aes = accel_dev->hw_device->num_engines;
60*bec61a29SDamian Muszynski 	const size_t hb_struct_size = sizeof(struct hb_cnt_pair);
61*bec61a29SDamian Muszynski 	const size_t exp_diff_size = array3_size(ADF_NUM_PKE_STRAND, max_aes,
62*bec61a29SDamian Muszynski 						 hb_struct_size);
63*bec61a29SDamian Muszynski 	const size_t dev_ctrs = size_mul(max_aes, hb_ctrs);
64*bec61a29SDamian Muszynski 	const size_t stats_size = size_mul(dev_ctrs, hb_struct_size);
65*bec61a29SDamian Muszynski 	const u32 exp_diff_cnt = exp_diff_size / sizeof(u32);
66*bec61a29SDamian Muszynski 	const u32 stats_el_cnt = stats_size / sizeof(u32);
67*bec61a29SDamian Muszynski 	struct hb_cnt_pair *hb_stats = accel_dev->heartbeat->dma.virt_addr;
68*bec61a29SDamian Muszynski 	const u32 *mem_to_chk = (u32 *)(hb_stats + dev_ctrs);
69*bec61a29SDamian Muszynski 	u32 el_diff_cnt = 0;
70*bec61a29SDamian Muszynski 	int i;
71*bec61a29SDamian Muszynski 
72*bec61a29SDamian Muszynski 	/* count how many bytes are different from pattern */
73*bec61a29SDamian Muszynski 	for (i = 0; i < stats_el_cnt; i++) {
74*bec61a29SDamian Muszynski 		if (mem_to_chk[i] == ADF_HB_EMPTY_SIG)
75*bec61a29SDamian Muszynski 			break;
76*bec61a29SDamian Muszynski 
77*bec61a29SDamian Muszynski 		el_diff_cnt++;
78*bec61a29SDamian Muszynski 	}
79*bec61a29SDamian Muszynski 
80*bec61a29SDamian Muszynski 	return el_diff_cnt && el_diff_cnt == exp_diff_cnt;
81*bec61a29SDamian Muszynski }
82*bec61a29SDamian Muszynski 
adf_heartbeat_check_ctrs(struct adf_accel_dev * accel_dev)83*bec61a29SDamian Muszynski void adf_heartbeat_check_ctrs(struct adf_accel_dev *accel_dev)
84*bec61a29SDamian Muszynski {
85*bec61a29SDamian Muszynski 	struct hb_cnt_pair *hb_stats = accel_dev->heartbeat->dma.virt_addr;
86*bec61a29SDamian Muszynski 	const size_t hb_ctrs = accel_dev->hw_device->num_hb_ctrs;
87*bec61a29SDamian Muszynski 	const size_t max_aes = accel_dev->hw_device->num_engines;
88*bec61a29SDamian Muszynski 	const size_t dev_ctrs = size_mul(max_aes, hb_ctrs);
89*bec61a29SDamian Muszynski 	const size_t stats_size = size_mul(dev_ctrs, sizeof(struct hb_cnt_pair));
90*bec61a29SDamian Muszynski 	const size_t mem_items_to_fill = size_mul(stats_size, 2) / sizeof(u32);
91*bec61a29SDamian Muszynski 
92*bec61a29SDamian Muszynski 	/* fill hb stats memory with pattern */
93*bec61a29SDamian Muszynski 	memset32((uint32_t *)hb_stats, ADF_HB_EMPTY_SIG, mem_items_to_fill);
94*bec61a29SDamian Muszynski 	accel_dev->heartbeat->ctrs_cnt_checked = false;
95*bec61a29SDamian Muszynski }
96*bec61a29SDamian Muszynski EXPORT_SYMBOL_GPL(adf_heartbeat_check_ctrs);
97*bec61a29SDamian Muszynski 
get_timer_ticks(struct adf_accel_dev * accel_dev,unsigned int * value)98359b84f8SDamian Muszynski static int get_timer_ticks(struct adf_accel_dev *accel_dev, unsigned int *value)
99359b84f8SDamian Muszynski {
100359b84f8SDamian Muszynski 	char timer_str[ADF_CFG_MAX_VAL_LEN_IN_BYTES] = { };
101359b84f8SDamian Muszynski 	u32 timer_ms = ADF_CFG_HB_TIMER_DEFAULT_MS;
102359b84f8SDamian Muszynski 	int cfg_read_status;
103359b84f8SDamian Muszynski 	u32 ticks;
104359b84f8SDamian Muszynski 	int ret;
105359b84f8SDamian Muszynski 
106359b84f8SDamian Muszynski 	cfg_read_status = adf_cfg_get_param_value(accel_dev, ADF_GENERAL_SEC,
107359b84f8SDamian Muszynski 						  ADF_HEARTBEAT_TIMER, timer_str);
108359b84f8SDamian Muszynski 	if (cfg_read_status == 0) {
109359b84f8SDamian Muszynski 		if (kstrtouint(timer_str, 10, &timer_ms))
110359b84f8SDamian Muszynski 			dev_dbg(&GET_DEV(accel_dev),
111359b84f8SDamian Muszynski 				"kstrtouint failed to parse the %s, param value",
112359b84f8SDamian Muszynski 				ADF_HEARTBEAT_TIMER);
113359b84f8SDamian Muszynski 	}
114359b84f8SDamian Muszynski 
115359b84f8SDamian Muszynski 	if (timer_ms < ADF_CFG_HB_TIMER_MIN_MS) {
116359b84f8SDamian Muszynski 		dev_err(&GET_DEV(accel_dev), "Timer cannot be less than %u\n",
117359b84f8SDamian Muszynski 			ADF_CFG_HB_TIMER_MIN_MS);
118359b84f8SDamian Muszynski 		return -EINVAL;
119359b84f8SDamian Muszynski 	}
120359b84f8SDamian Muszynski 
121359b84f8SDamian Muszynski 	/*
122359b84f8SDamian Muszynski 	 * On 4xxx devices adf_timer is responsible for HB updates and
123359b84f8SDamian Muszynski 	 * its period is fixed to 200ms
124359b84f8SDamian Muszynski 	 */
125359b84f8SDamian Muszynski 	if (accel_dev->timer)
126359b84f8SDamian Muszynski 		timer_ms = ADF_CFG_HB_TIMER_MIN_MS;
127359b84f8SDamian Muszynski 
128359b84f8SDamian Muszynski 	ret = adf_heartbeat_ms_to_ticks(accel_dev, timer_ms, &ticks);
129359b84f8SDamian Muszynski 	if (ret)
130359b84f8SDamian Muszynski 		return ret;
131359b84f8SDamian Muszynski 
132359b84f8SDamian Muszynski 	adf_heartbeat_save_cfg_param(accel_dev, timer_ms);
133359b84f8SDamian Muszynski 
134359b84f8SDamian Muszynski 	accel_dev->heartbeat->hb_timer = timer_ms;
135359b84f8SDamian Muszynski 	*value = ticks;
136359b84f8SDamian Muszynski 
137359b84f8SDamian Muszynski 	return 0;
138359b84f8SDamian Muszynski }
139359b84f8SDamian Muszynski 
check_ae(struct hb_cnt_pair * curr,struct hb_cnt_pair * prev,u16 * count,const size_t hb_ctrs)140359b84f8SDamian Muszynski static int check_ae(struct hb_cnt_pair *curr, struct hb_cnt_pair *prev,
141359b84f8SDamian Muszynski 		    u16 *count, const size_t hb_ctrs)
142359b84f8SDamian Muszynski {
143359b84f8SDamian Muszynski 	size_t thr;
144359b84f8SDamian Muszynski 
145359b84f8SDamian Muszynski 	/* loop through all threads in AE */
146359b84f8SDamian Muszynski 	for (thr = 0; thr < hb_ctrs; thr++) {
147359b84f8SDamian Muszynski 		u16 req = curr[thr].req_heartbeat_cnt;
148359b84f8SDamian Muszynski 		u16 resp = curr[thr].resp_heartbeat_cnt;
149359b84f8SDamian Muszynski 		u16 last = prev[thr].resp_heartbeat_cnt;
150359b84f8SDamian Muszynski 
151359b84f8SDamian Muszynski 		if ((thr == ADF_AE_ADMIN_THREAD || req != resp) && resp == last) {
152359b84f8SDamian Muszynski 			u16 retry = ++count[thr];
153359b84f8SDamian Muszynski 
154359b84f8SDamian Muszynski 			if (retry >= ADF_CFG_HB_COUNT_THRESHOLD)
155359b84f8SDamian Muszynski 				return -EIO;
156359b84f8SDamian Muszynski 
157359b84f8SDamian Muszynski 		} else {
158359b84f8SDamian Muszynski 			count[thr] = 0;
159359b84f8SDamian Muszynski 		}
160359b84f8SDamian Muszynski 	}
161359b84f8SDamian Muszynski 	return 0;
162359b84f8SDamian Muszynski }
163359b84f8SDamian Muszynski 
adf_hb_get_status(struct adf_accel_dev * accel_dev)164359b84f8SDamian Muszynski static int adf_hb_get_status(struct adf_accel_dev *accel_dev)
165359b84f8SDamian Muszynski {
166359b84f8SDamian Muszynski 	struct adf_hw_device_data *hw_device = accel_dev->hw_device;
167359b84f8SDamian Muszynski 	struct hb_cnt_pair *live_stats, *last_stats, *curr_stats;
168359b84f8SDamian Muszynski 	const size_t hb_ctrs = hw_device->num_hb_ctrs;
169359b84f8SDamian Muszynski 	const unsigned long ae_mask = hw_device->ae_mask;
170359b84f8SDamian Muszynski 	const size_t max_aes = hw_device->num_engines;
171359b84f8SDamian Muszynski 	const size_t dev_ctrs = size_mul(max_aes, hb_ctrs);
172359b84f8SDamian Muszynski 	const size_t stats_size = size_mul(dev_ctrs, sizeof(*curr_stats));
173359b84f8SDamian Muszynski 	struct hb_cnt_pair *ae_curr_p, *ae_prev_p;
174359b84f8SDamian Muszynski 	u16 *count_fails, *ae_count_p;
175359b84f8SDamian Muszynski 	size_t ae_offset;
176359b84f8SDamian Muszynski 	size_t ae = 0;
177359b84f8SDamian Muszynski 	int ret = 0;
178359b84f8SDamian Muszynski 
179*bec61a29SDamian Muszynski 	if (!accel_dev->heartbeat->ctrs_cnt_checked) {
180*bec61a29SDamian Muszynski 		if (validate_hb_ctrs_cnt(accel_dev))
181*bec61a29SDamian Muszynski 			hw_device->num_hb_ctrs += ADF_NUM_PKE_STRAND;
182*bec61a29SDamian Muszynski 
183*bec61a29SDamian Muszynski 		accel_dev->heartbeat->ctrs_cnt_checked = true;
184*bec61a29SDamian Muszynski 	}
185*bec61a29SDamian Muszynski 
186359b84f8SDamian Muszynski 	live_stats = accel_dev->heartbeat->dma.virt_addr;
187359b84f8SDamian Muszynski 	last_stats = live_stats + dev_ctrs;
188359b84f8SDamian Muszynski 	count_fails = (u16 *)(last_stats + dev_ctrs);
189359b84f8SDamian Muszynski 
190359b84f8SDamian Muszynski 	curr_stats = kmemdup(live_stats, stats_size, GFP_KERNEL);
191359b84f8SDamian Muszynski 	if (!curr_stats)
192359b84f8SDamian Muszynski 		return -ENOMEM;
193359b84f8SDamian Muszynski 
194359b84f8SDamian Muszynski 	/* loop through active AEs */
195359b84f8SDamian Muszynski 	for_each_set_bit(ae, &ae_mask, max_aes) {
196359b84f8SDamian Muszynski 		ae_offset = size_mul(ae, hb_ctrs);
197359b84f8SDamian Muszynski 		ae_curr_p = curr_stats + ae_offset;
198359b84f8SDamian Muszynski 		ae_prev_p = last_stats + ae_offset;
199359b84f8SDamian Muszynski 		ae_count_p = count_fails + ae_offset;
200359b84f8SDamian Muszynski 
201359b84f8SDamian Muszynski 		ret = check_ae(ae_curr_p, ae_prev_p, ae_count_p, hb_ctrs);
202359b84f8SDamian Muszynski 		if (ret)
203359b84f8SDamian Muszynski 			break;
204359b84f8SDamian Muszynski 	}
205359b84f8SDamian Muszynski 
206359b84f8SDamian Muszynski 	/* Copy current stats for the next iteration */
207359b84f8SDamian Muszynski 	memcpy(last_stats, curr_stats, stats_size);
208359b84f8SDamian Muszynski 	kfree(curr_stats);
209359b84f8SDamian Muszynski 
210359b84f8SDamian Muszynski 	return ret;
211359b84f8SDamian Muszynski }
212359b84f8SDamian Muszynski 
adf_heartbeat_status(struct adf_accel_dev * accel_dev,enum adf_device_heartbeat_status * hb_status)213359b84f8SDamian Muszynski void adf_heartbeat_status(struct adf_accel_dev *accel_dev,
214359b84f8SDamian Muszynski 			  enum adf_device_heartbeat_status *hb_status)
215359b84f8SDamian Muszynski {
216359b84f8SDamian Muszynski 	struct adf_heartbeat *hb;
217359b84f8SDamian Muszynski 
218359b84f8SDamian Muszynski 	if (!adf_dev_started(accel_dev) ||
219359b84f8SDamian Muszynski 	    test_bit(ADF_STATUS_RESTARTING, &accel_dev->status)) {
220359b84f8SDamian Muszynski 		*hb_status = HB_DEV_UNRESPONSIVE;
221359b84f8SDamian Muszynski 		return;
222359b84f8SDamian Muszynski 	}
223359b84f8SDamian Muszynski 
224359b84f8SDamian Muszynski 	if (adf_hb_check_polling_freq(accel_dev) == -EINVAL) {
225359b84f8SDamian Muszynski 		*hb_status = HB_DEV_UNSUPPORTED;
226359b84f8SDamian Muszynski 		return;
227359b84f8SDamian Muszynski 	}
228359b84f8SDamian Muszynski 
229359b84f8SDamian Muszynski 	hb = accel_dev->heartbeat;
230359b84f8SDamian Muszynski 	hb->hb_sent_counter++;
231359b84f8SDamian Muszynski 
232359b84f8SDamian Muszynski 	if (adf_hb_get_status(accel_dev)) {
233359b84f8SDamian Muszynski 		dev_err(&GET_DEV(accel_dev),
234359b84f8SDamian Muszynski 			"Heartbeat ERROR: QAT is not responding.\n");
235359b84f8SDamian Muszynski 		*hb_status = HB_DEV_UNRESPONSIVE;
236359b84f8SDamian Muszynski 		hb->hb_failed_counter++;
237359b84f8SDamian Muszynski 		return;
238359b84f8SDamian Muszynski 	}
239359b84f8SDamian Muszynski 
240359b84f8SDamian Muszynski 	*hb_status = HB_DEV_ALIVE;
241359b84f8SDamian Muszynski }
242359b84f8SDamian Muszynski 
adf_heartbeat_ms_to_ticks(struct adf_accel_dev * accel_dev,unsigned int time_ms,u32 * value)243359b84f8SDamian Muszynski int adf_heartbeat_ms_to_ticks(struct adf_accel_dev *accel_dev, unsigned int time_ms,
244359b84f8SDamian Muszynski 			      u32 *value)
245359b84f8SDamian Muszynski {
246359b84f8SDamian Muszynski 	struct adf_hw_device_data *hw_data = accel_dev->hw_device;
247359b84f8SDamian Muszynski 	u32 clk_per_sec;
248359b84f8SDamian Muszynski 
249359b84f8SDamian Muszynski 	/* HB clock may be different than AE clock */
250359b84f8SDamian Muszynski 	if (!hw_data->get_hb_clock)
251359b84f8SDamian Muszynski 		return -EINVAL;
252359b84f8SDamian Muszynski 
253359b84f8SDamian Muszynski 	clk_per_sec = hw_data->get_hb_clock(hw_data);
254359b84f8SDamian Muszynski 	*value = time_ms * (clk_per_sec / MSEC_PER_SEC);
255359b84f8SDamian Muszynski 
256359b84f8SDamian Muszynski 	return 0;
257359b84f8SDamian Muszynski }
258359b84f8SDamian Muszynski 
adf_heartbeat_save_cfg_param(struct adf_accel_dev * accel_dev,unsigned int timer_ms)259359b84f8SDamian Muszynski int adf_heartbeat_save_cfg_param(struct adf_accel_dev *accel_dev,
260359b84f8SDamian Muszynski 				 unsigned int timer_ms)
261359b84f8SDamian Muszynski {
262359b84f8SDamian Muszynski 	char timer_str[ADF_CFG_MAX_VAL_LEN_IN_BYTES];
263359b84f8SDamian Muszynski 
264359b84f8SDamian Muszynski 	snprintf(timer_str, sizeof(timer_str), "%u", timer_ms);
265359b84f8SDamian Muszynski 	return adf_cfg_add_key_value_param(accel_dev, ADF_GENERAL_SEC,
266359b84f8SDamian Muszynski 					  ADF_HEARTBEAT_TIMER, timer_str,
267359b84f8SDamian Muszynski 					  ADF_STR);
268359b84f8SDamian Muszynski }
269359b84f8SDamian Muszynski EXPORT_SYMBOL_GPL(adf_heartbeat_save_cfg_param);
270359b84f8SDamian Muszynski 
adf_heartbeat_init(struct adf_accel_dev * accel_dev)271359b84f8SDamian Muszynski int adf_heartbeat_init(struct adf_accel_dev *accel_dev)
272359b84f8SDamian Muszynski {
273359b84f8SDamian Muszynski 	struct adf_heartbeat *hb;
274359b84f8SDamian Muszynski 
275359b84f8SDamian Muszynski 	hb = kzalloc(sizeof(*hb), GFP_KERNEL);
276359b84f8SDamian Muszynski 	if (!hb)
277359b84f8SDamian Muszynski 		goto err_ret;
278359b84f8SDamian Muszynski 
279359b84f8SDamian Muszynski 	hb->dma.virt_addr = dma_alloc_coherent(&GET_DEV(accel_dev), PAGE_SIZE,
280359b84f8SDamian Muszynski 					       &hb->dma.phy_addr, GFP_KERNEL);
281359b84f8SDamian Muszynski 	if (!hb->dma.virt_addr)
282359b84f8SDamian Muszynski 		goto err_free;
283359b84f8SDamian Muszynski 
284*bec61a29SDamian Muszynski 	/*
285*bec61a29SDamian Muszynski 	 * Default set this flag as true to avoid unnecessary checks,
286*bec61a29SDamian Muszynski 	 * it will be reset on platforms that need such a check
287*bec61a29SDamian Muszynski 	 */
288*bec61a29SDamian Muszynski 	hb->ctrs_cnt_checked = true;
289359b84f8SDamian Muszynski 	accel_dev->heartbeat = hb;
290359b84f8SDamian Muszynski 
291359b84f8SDamian Muszynski 	return 0;
292359b84f8SDamian Muszynski 
293359b84f8SDamian Muszynski err_free:
294359b84f8SDamian Muszynski 	kfree(hb);
295359b84f8SDamian Muszynski err_ret:
296359b84f8SDamian Muszynski 	return -ENOMEM;
297359b84f8SDamian Muszynski }
298359b84f8SDamian Muszynski 
adf_heartbeat_start(struct adf_accel_dev * accel_dev)299359b84f8SDamian Muszynski int adf_heartbeat_start(struct adf_accel_dev *accel_dev)
300359b84f8SDamian Muszynski {
301359b84f8SDamian Muszynski 	unsigned int timer_ticks;
302359b84f8SDamian Muszynski 	int ret;
303359b84f8SDamian Muszynski 
304359b84f8SDamian Muszynski 	if (!accel_dev->heartbeat) {
305359b84f8SDamian Muszynski 		dev_warn(&GET_DEV(accel_dev), "Heartbeat instance not found!");
306359b84f8SDamian Muszynski 		return -EFAULT;
307359b84f8SDamian Muszynski 	}
308359b84f8SDamian Muszynski 
309*bec61a29SDamian Muszynski 	if (accel_dev->hw_device->check_hb_ctrs)
310*bec61a29SDamian Muszynski 		accel_dev->hw_device->check_hb_ctrs(accel_dev);
311*bec61a29SDamian Muszynski 
312359b84f8SDamian Muszynski 	ret = get_timer_ticks(accel_dev, &timer_ticks);
313359b84f8SDamian Muszynski 	if (ret)
314359b84f8SDamian Muszynski 		return ret;
315359b84f8SDamian Muszynski 
316359b84f8SDamian Muszynski 	ret = adf_send_admin_hb_timer(accel_dev, timer_ticks);
317359b84f8SDamian Muszynski 	if (ret)
318359b84f8SDamian Muszynski 		dev_warn(&GET_DEV(accel_dev), "Heartbeat not supported!");
319359b84f8SDamian Muszynski 
320359b84f8SDamian Muszynski 	return ret;
321359b84f8SDamian Muszynski }
322359b84f8SDamian Muszynski 
adf_heartbeat_shutdown(struct adf_accel_dev * accel_dev)323359b84f8SDamian Muszynski void adf_heartbeat_shutdown(struct adf_accel_dev *accel_dev)
324359b84f8SDamian Muszynski {
325359b84f8SDamian Muszynski 	struct adf_heartbeat *hb = accel_dev->heartbeat;
326359b84f8SDamian Muszynski 
327359b84f8SDamian Muszynski 	if (!hb)
328359b84f8SDamian Muszynski 		return;
329359b84f8SDamian Muszynski 
330359b84f8SDamian Muszynski 	if (hb->dma.virt_addr)
331359b84f8SDamian Muszynski 		dma_free_coherent(&GET_DEV(accel_dev), PAGE_SIZE,
332359b84f8SDamian Muszynski 				  hb->dma.virt_addr, hb->dma.phy_addr);
333359b84f8SDamian Muszynski 
334359b84f8SDamian Muszynski 	kfree(hb);
335359b84f8SDamian Muszynski 	accel_dev->heartbeat = NULL;
336359b84f8SDamian Muszynski }
337