1 // SPDX-License-Identifier: GPL-2.0-only
2 //
3 // KUnit test for the Cirrus common amplifier library.
4 //
5 // Copyright (C) 2024 Cirrus Logic, Inc. and
6 // Cirrus Logic International Semiconductor Ltd.
7
8 #include <kunit/test.h>
9 #include <kunit/static_stub.h>
10 #include <linux/firmware/cirrus/cs_dsp.h>
11 #include <linux/firmware/cirrus/wmfw.h>
12 #include <linux/gpio/driver.h>
13 #include <linux/list.h>
14 #include <linux/module.h>
15 #include <linux/platform_device.h>
16 #include <linux/random.h>
17 #include <sound/cs-amp-lib.h>
18
19 struct cs_amp_lib_test_priv {
20 struct platform_device amp_pdev;
21
22 struct cirrus_amp_efi_data *cal_blob;
23 struct list_head ctl_write_list;
24 };
25
26 struct cs_amp_lib_test_ctl_write_entry {
27 struct list_head list;
28 unsigned int value;
29 char name[16];
30 };
31
32 struct cs_amp_lib_test_param {
33 int num_amps;
34 int amp_index;
35 };
36
cs_amp_lib_test_init_dummy_cal_blob(struct kunit * test,int num_amps)37 static void cs_amp_lib_test_init_dummy_cal_blob(struct kunit *test, int num_amps)
38 {
39 struct cs_amp_lib_test_priv *priv = test->priv;
40 unsigned int blob_size;
41 int i;
42
43 blob_size = offsetof(struct cirrus_amp_efi_data, data) +
44 sizeof(struct cirrus_amp_cal_data) * num_amps;
45
46 priv->cal_blob = kunit_kzalloc(test, blob_size, GFP_KERNEL);
47 KUNIT_ASSERT_NOT_NULL(test, priv->cal_blob);
48
49 priv->cal_blob->size = blob_size;
50 priv->cal_blob->count = num_amps;
51
52 get_random_bytes(priv->cal_blob->data, sizeof(struct cirrus_amp_cal_data) * num_amps);
53
54 /* Ensure all timestamps are non-zero to mark the entry valid. */
55 for (i = 0; i < num_amps; i++)
56 priv->cal_blob->data[i].calTime[0] |= 1;
57
58 /* Ensure that all UIDs are non-zero and unique. */
59 for (i = 0; i < num_amps; i++)
60 *(u8 *)&priv->cal_blob->data[i].calTarget[0] = i + 1;
61 }
62
cs_amp_lib_test_get_target_uid(struct kunit * test)63 static u64 cs_amp_lib_test_get_target_uid(struct kunit *test)
64 {
65 struct cs_amp_lib_test_priv *priv = test->priv;
66 const struct cs_amp_lib_test_param *param = test->param_value;
67 u64 uid;
68
69 uid = priv->cal_blob->data[param->amp_index].calTarget[1];
70 uid <<= 32;
71 uid |= priv->cal_blob->data[param->amp_index].calTarget[0];
72
73 return uid;
74 }
75
76 /* Redirected get_efi_variable to simulate that the file is too short */
cs_amp_lib_test_get_efi_variable_nohead(efi_char16_t * name,efi_guid_t * guid,unsigned long * size,void * buf)77 static efi_status_t cs_amp_lib_test_get_efi_variable_nohead(efi_char16_t *name,
78 efi_guid_t *guid,
79 unsigned long *size,
80 void *buf)
81 {
82 if (!buf) {
83 *size = offsetof(struct cirrus_amp_efi_data, data) - 1;
84 return EFI_BUFFER_TOO_SMALL;
85 }
86
87 return EFI_NOT_FOUND;
88 }
89
90 /* Should return -EOVERFLOW if the header is larger than the EFI data */
cs_amp_lib_test_cal_data_too_short_test(struct kunit * test)91 static void cs_amp_lib_test_cal_data_too_short_test(struct kunit *test)
92 {
93 struct cs_amp_lib_test_priv *priv = test->priv;
94 struct cirrus_amp_cal_data result_data;
95 int ret;
96
97 /* Redirect calls to get EFI data */
98 kunit_activate_static_stub(test,
99 cs_amp_test_hooks->get_efi_variable,
100 cs_amp_lib_test_get_efi_variable_nohead);
101
102 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, 0, 0, &result_data);
103 KUNIT_EXPECT_EQ(test, ret, -EOVERFLOW);
104
105 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
106 }
107
108 /* Redirected get_efi_variable to simulate that the count is larger than the file */
cs_amp_lib_test_get_efi_variable_bad_count(efi_char16_t * name,efi_guid_t * guid,unsigned long * size,void * buf)109 static efi_status_t cs_amp_lib_test_get_efi_variable_bad_count(efi_char16_t *name,
110 efi_guid_t *guid,
111 unsigned long *size,
112 void *buf)
113 {
114 struct kunit *test = kunit_get_current_test();
115 struct cs_amp_lib_test_priv *priv = test->priv;
116
117 if (!buf) {
118 /*
119 * Return a size that is shorter than required for the
120 * declared number of entries.
121 */
122 *size = priv->cal_blob->size - 1;
123 return EFI_BUFFER_TOO_SMALL;
124 }
125
126 memcpy(buf, priv->cal_blob, priv->cal_blob->size - 1);
127
128 return EFI_SUCCESS;
129 }
130
131 /* Should return -EOVERFLOW if the entry count is larger than the EFI data */
cs_amp_lib_test_cal_count_too_big_test(struct kunit * test)132 static void cs_amp_lib_test_cal_count_too_big_test(struct kunit *test)
133 {
134 struct cs_amp_lib_test_priv *priv = test->priv;
135 struct cirrus_amp_cal_data result_data;
136 int ret;
137
138 cs_amp_lib_test_init_dummy_cal_blob(test, 8);
139
140 /* Redirect calls to get EFI data */
141 kunit_activate_static_stub(test,
142 cs_amp_test_hooks->get_efi_variable,
143 cs_amp_lib_test_get_efi_variable_bad_count);
144
145 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, 0, 0, &result_data);
146 KUNIT_EXPECT_EQ(test, ret, -EOVERFLOW);
147
148 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
149 }
150
151 /* Redirected get_efi_variable to simulate that the variable not found */
cs_amp_lib_test_get_efi_variable_none(efi_char16_t * name,efi_guid_t * guid,unsigned long * size,void * buf)152 static efi_status_t cs_amp_lib_test_get_efi_variable_none(efi_char16_t *name,
153 efi_guid_t *guid,
154 unsigned long *size,
155 void *buf)
156 {
157 return EFI_NOT_FOUND;
158 }
159
160 /* If EFI doesn't contain a cal data variable the result should be -ENOENT */
cs_amp_lib_test_no_cal_data_test(struct kunit * test)161 static void cs_amp_lib_test_no_cal_data_test(struct kunit *test)
162 {
163 struct cs_amp_lib_test_priv *priv = test->priv;
164 struct cirrus_amp_cal_data result_data;
165 int ret;
166
167 /* Redirect calls to get EFI data */
168 kunit_activate_static_stub(test,
169 cs_amp_test_hooks->get_efi_variable,
170 cs_amp_lib_test_get_efi_variable_none);
171
172 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, 0, 0, &result_data);
173 KUNIT_EXPECT_EQ(test, ret, -ENOENT);
174
175 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
176 }
177
178 /* Redirected get_efi_variable to simulate reading a cal data blob */
cs_amp_lib_test_get_efi_variable(efi_char16_t * name,efi_guid_t * guid,unsigned long * size,void * buf)179 static efi_status_t cs_amp_lib_test_get_efi_variable(efi_char16_t *name,
180 efi_guid_t *guid,
181 unsigned long *size,
182 void *buf)
183 {
184 static const efi_char16_t expected_name[] = L"CirrusSmartAmpCalibrationData";
185 static const efi_guid_t expected_guid =
186 EFI_GUID(0x02f9af02, 0x7734, 0x4233, 0xb4, 0x3d, 0x93, 0xfe, 0x5a, 0xa3, 0x5d, 0xb3);
187 struct kunit *test = kunit_get_current_test();
188 struct cs_amp_lib_test_priv *priv = test->priv;
189
190 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, name);
191 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, guid);
192 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, size);
193
194 KUNIT_EXPECT_MEMEQ(test, name, expected_name, sizeof(expected_name));
195 KUNIT_EXPECT_MEMEQ(test, guid, &expected_guid, sizeof(expected_guid));
196
197 if (!buf) {
198 *size = priv->cal_blob->size;
199 return EFI_BUFFER_TOO_SMALL;
200 }
201
202 KUNIT_ASSERT_GE_MSG(test, ksize(buf), priv->cal_blob->size, "Buffer to small");
203
204 memcpy(buf, priv->cal_blob, priv->cal_blob->size);
205
206 return EFI_SUCCESS;
207 }
208
209 /* Get cal data block for a given amp, matched by target UID. */
cs_amp_lib_test_get_efi_cal_by_uid_test(struct kunit * test)210 static void cs_amp_lib_test_get_efi_cal_by_uid_test(struct kunit *test)
211 {
212 struct cs_amp_lib_test_priv *priv = test->priv;
213 const struct cs_amp_lib_test_param *param = test->param_value;
214 struct cirrus_amp_cal_data result_data;
215 u64 target_uid;
216 int ret;
217
218 cs_amp_lib_test_init_dummy_cal_blob(test, param->num_amps);
219
220 /* Redirect calls to get EFI data */
221 kunit_activate_static_stub(test,
222 cs_amp_test_hooks->get_efi_variable,
223 cs_amp_lib_test_get_efi_variable);
224
225 target_uid = cs_amp_lib_test_get_target_uid(test);
226 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, target_uid, -1, &result_data);
227 KUNIT_EXPECT_EQ(test, ret, 0);
228
229 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
230
231 KUNIT_EXPECT_EQ(test, result_data.calTarget[0], target_uid & 0xFFFFFFFFULL);
232 KUNIT_EXPECT_EQ(test, result_data.calTarget[1], target_uid >> 32);
233 KUNIT_EXPECT_EQ(test, result_data.calTime[0],
234 priv->cal_blob->data[param->amp_index].calTime[0]);
235 KUNIT_EXPECT_EQ(test, result_data.calTime[1],
236 priv->cal_blob->data[param->amp_index].calTime[1]);
237 KUNIT_EXPECT_EQ(test, result_data.calAmbient,
238 priv->cal_blob->data[param->amp_index].calAmbient);
239 KUNIT_EXPECT_EQ(test, result_data.calStatus,
240 priv->cal_blob->data[param->amp_index].calStatus);
241 KUNIT_EXPECT_EQ(test, result_data.calR,
242 priv->cal_blob->data[param->amp_index].calR);
243 }
244
245 /* Get cal data block for a given amp index without checking target UID. */
cs_amp_lib_test_get_efi_cal_by_index_unchecked_test(struct kunit * test)246 static void cs_amp_lib_test_get_efi_cal_by_index_unchecked_test(struct kunit *test)
247 {
248 struct cs_amp_lib_test_priv *priv = test->priv;
249 const struct cs_amp_lib_test_param *param = test->param_value;
250 struct cirrus_amp_cal_data result_data;
251 int ret;
252
253 cs_amp_lib_test_init_dummy_cal_blob(test, param->num_amps);
254
255 /* Redirect calls to get EFI data */
256 kunit_activate_static_stub(test,
257 cs_amp_test_hooks->get_efi_variable,
258 cs_amp_lib_test_get_efi_variable);
259
260 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, 0,
261 param->amp_index, &result_data);
262 KUNIT_EXPECT_EQ(test, ret, 0);
263
264 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
265
266 KUNIT_EXPECT_EQ(test, result_data.calTime[0],
267 priv->cal_blob->data[param->amp_index].calTime[0]);
268 KUNIT_EXPECT_EQ(test, result_data.calTime[1],
269 priv->cal_blob->data[param->amp_index].calTime[1]);
270 KUNIT_EXPECT_EQ(test, result_data.calAmbient,
271 priv->cal_blob->data[param->amp_index].calAmbient);
272 KUNIT_EXPECT_EQ(test, result_data.calStatus,
273 priv->cal_blob->data[param->amp_index].calStatus);
274 KUNIT_EXPECT_EQ(test, result_data.calR,
275 priv->cal_blob->data[param->amp_index].calR);
276 }
277
278 /* Get cal data block for a given amp index with checked target UID. */
cs_amp_lib_test_get_efi_cal_by_index_checked_test(struct kunit * test)279 static void cs_amp_lib_test_get_efi_cal_by_index_checked_test(struct kunit *test)
280 {
281 struct cs_amp_lib_test_priv *priv = test->priv;
282 const struct cs_amp_lib_test_param *param = test->param_value;
283 struct cirrus_amp_cal_data result_data;
284 u64 target_uid;
285 int ret;
286
287 cs_amp_lib_test_init_dummy_cal_blob(test, param->num_amps);
288
289 /* Redirect calls to get EFI data */
290 kunit_activate_static_stub(test,
291 cs_amp_test_hooks->get_efi_variable,
292 cs_amp_lib_test_get_efi_variable);
293
294 target_uid = cs_amp_lib_test_get_target_uid(test);
295 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, target_uid,
296 param->amp_index, &result_data);
297 KUNIT_EXPECT_EQ(test, ret, 0);
298
299 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
300
301 KUNIT_EXPECT_EQ(test, result_data.calTime[0],
302 priv->cal_blob->data[param->amp_index].calTime[0]);
303 KUNIT_EXPECT_EQ(test, result_data.calTime[1],
304 priv->cal_blob->data[param->amp_index].calTime[1]);
305 KUNIT_EXPECT_EQ(test, result_data.calAmbient,
306 priv->cal_blob->data[param->amp_index].calAmbient);
307 KUNIT_EXPECT_EQ(test, result_data.calStatus,
308 priv->cal_blob->data[param->amp_index].calStatus);
309 KUNIT_EXPECT_EQ(test, result_data.calR,
310 priv->cal_blob->data[param->amp_index].calR);
311 }
312
313 /*
314 * Get cal data block for a given amp index with checked target UID.
315 * The UID does not match so the result should be -ENOENT.
316 */
cs_amp_lib_test_get_efi_cal_by_index_uid_mismatch_test(struct kunit * test)317 static void cs_amp_lib_test_get_efi_cal_by_index_uid_mismatch_test(struct kunit *test)
318 {
319 struct cs_amp_lib_test_priv *priv = test->priv;
320 const struct cs_amp_lib_test_param *param = test->param_value;
321 struct cirrus_amp_cal_data result_data;
322 u64 target_uid;
323 int ret;
324
325 cs_amp_lib_test_init_dummy_cal_blob(test, param->num_amps);
326
327 /* Redirect calls to get EFI data */
328 kunit_activate_static_stub(test,
329 cs_amp_test_hooks->get_efi_variable,
330 cs_amp_lib_test_get_efi_variable);
331
332 /* Get a target UID that won't match the entry */
333 target_uid = ~cs_amp_lib_test_get_target_uid(test);
334 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, target_uid,
335 param->amp_index, &result_data);
336 KUNIT_EXPECT_EQ(test, ret, -ENOENT);
337
338 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
339 }
340
341 /*
342 * Get cal data block for a given amp, where the cal data does not
343 * specify calTarget so the lookup falls back to using the index
344 */
cs_amp_lib_test_get_efi_cal_by_index_fallback_test(struct kunit * test)345 static void cs_amp_lib_test_get_efi_cal_by_index_fallback_test(struct kunit *test)
346 {
347 struct cs_amp_lib_test_priv *priv = test->priv;
348 const struct cs_amp_lib_test_param *param = test->param_value;
349 struct cirrus_amp_cal_data result_data;
350 static const u64 bad_target_uid = 0xBADCA100BABABABAULL;
351 int i, ret;
352
353 cs_amp_lib_test_init_dummy_cal_blob(test, param->num_amps);
354
355 /* Make all the target values zero so they are ignored */
356 for (i = 0; i < priv->cal_blob->count; ++i) {
357 priv->cal_blob->data[i].calTarget[0] = 0;
358 priv->cal_blob->data[i].calTarget[1] = 0;
359 }
360
361 /* Redirect calls to get EFI data */
362 kunit_activate_static_stub(test,
363 cs_amp_test_hooks->get_efi_variable,
364 cs_amp_lib_test_get_efi_variable);
365
366 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, bad_target_uid,
367 param->amp_index, &result_data);
368 KUNIT_EXPECT_EQ(test, ret, 0);
369
370 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
371
372 KUNIT_EXPECT_EQ(test, result_data.calTime[0],
373 priv->cal_blob->data[param->amp_index].calTime[0]);
374 KUNIT_EXPECT_EQ(test, result_data.calTime[1],
375 priv->cal_blob->data[param->amp_index].calTime[1]);
376 KUNIT_EXPECT_EQ(test, result_data.calAmbient,
377 priv->cal_blob->data[param->amp_index].calAmbient);
378 KUNIT_EXPECT_EQ(test, result_data.calStatus,
379 priv->cal_blob->data[param->amp_index].calStatus);
380 KUNIT_EXPECT_EQ(test, result_data.calR,
381 priv->cal_blob->data[param->amp_index].calR);
382 }
383
384 /*
385 * If the target UID isn't present in the cal data, and there isn't an
386 * index to fall back do, the result should be -ENOENT.
387 */
cs_amp_lib_test_get_efi_cal_uid_not_found_noindex_test(struct kunit * test)388 static void cs_amp_lib_test_get_efi_cal_uid_not_found_noindex_test(struct kunit *test)
389 {
390 struct cs_amp_lib_test_priv *priv = test->priv;
391 struct cirrus_amp_cal_data result_data;
392 static const u64 bad_target_uid = 0xBADCA100BABABABAULL;
393 int i, ret;
394
395 cs_amp_lib_test_init_dummy_cal_blob(test, 8);
396
397 /* Make all the target values != bad_target_uid */
398 for (i = 0; i < priv->cal_blob->count; ++i) {
399 priv->cal_blob->data[i].calTarget[0] &= ~(bad_target_uid & 0xFFFFFFFFULL);
400 priv->cal_blob->data[i].calTarget[1] &= ~(bad_target_uid >> 32);
401 }
402
403 /* Redirect calls to get EFI data */
404 kunit_activate_static_stub(test,
405 cs_amp_test_hooks->get_efi_variable,
406 cs_amp_lib_test_get_efi_variable);
407
408 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, bad_target_uid, -1,
409 &result_data);
410 KUNIT_EXPECT_EQ(test, ret, -ENOENT);
411
412 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
413 }
414
415 /*
416 * If the target UID isn't present in the cal data, and the index is
417 * out of range, the result should be -ENOENT.
418 */
cs_amp_lib_test_get_efi_cal_uid_not_found_index_not_found_test(struct kunit * test)419 static void cs_amp_lib_test_get_efi_cal_uid_not_found_index_not_found_test(struct kunit *test)
420 {
421 struct cs_amp_lib_test_priv *priv = test->priv;
422 struct cirrus_amp_cal_data result_data;
423 static const u64 bad_target_uid = 0xBADCA100BABABABAULL;
424 int i, ret;
425
426 cs_amp_lib_test_init_dummy_cal_blob(test, 8);
427
428 /* Make all the target values != bad_target_uid */
429 for (i = 0; i < priv->cal_blob->count; ++i) {
430 priv->cal_blob->data[i].calTarget[0] &= ~(bad_target_uid & 0xFFFFFFFFULL);
431 priv->cal_blob->data[i].calTarget[1] &= ~(bad_target_uid >> 32);
432 }
433
434 /* Redirect calls to get EFI data */
435 kunit_activate_static_stub(test,
436 cs_amp_test_hooks->get_efi_variable,
437 cs_amp_lib_test_get_efi_variable);
438
439 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, bad_target_uid, 99,
440 &result_data);
441 KUNIT_EXPECT_EQ(test, ret, -ENOENT);
442
443 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
444 }
445
446 /*
447 * If the target UID isn't given, and the index is out of range, the
448 * result should be -ENOENT.
449 */
cs_amp_lib_test_get_efi_cal_no_uid_index_not_found_test(struct kunit * test)450 static void cs_amp_lib_test_get_efi_cal_no_uid_index_not_found_test(struct kunit *test)
451 {
452 struct cs_amp_lib_test_priv *priv = test->priv;
453 struct cirrus_amp_cal_data result_data;
454 int ret;
455
456 cs_amp_lib_test_init_dummy_cal_blob(test, 8);
457
458 /* Redirect calls to get EFI data */
459 kunit_activate_static_stub(test,
460 cs_amp_test_hooks->get_efi_variable,
461 cs_amp_lib_test_get_efi_variable);
462
463 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, 0, 99, &result_data);
464 KUNIT_EXPECT_EQ(test, ret, -ENOENT);
465
466 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
467 }
468
469 /* If neither the target UID or the index is given the result should be -ENOENT. */
cs_amp_lib_test_get_efi_cal_no_uid_no_index_test(struct kunit * test)470 static void cs_amp_lib_test_get_efi_cal_no_uid_no_index_test(struct kunit *test)
471 {
472 struct cs_amp_lib_test_priv *priv = test->priv;
473 struct cirrus_amp_cal_data result_data;
474 int ret;
475
476 cs_amp_lib_test_init_dummy_cal_blob(test, 8);
477
478 /* Redirect calls to get EFI data */
479 kunit_activate_static_stub(test,
480 cs_amp_test_hooks->get_efi_variable,
481 cs_amp_lib_test_get_efi_variable);
482
483 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, 0, -1, &result_data);
484 KUNIT_EXPECT_EQ(test, ret, -ENOENT);
485
486 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
487 }
488
489 /*
490 * If the UID is passed as 0 this must not match an entry with an
491 * unpopulated calTarget
492 */
cs_amp_lib_test_get_efi_cal_zero_not_matched_test(struct kunit * test)493 static void cs_amp_lib_test_get_efi_cal_zero_not_matched_test(struct kunit *test)
494 {
495 struct cs_amp_lib_test_priv *priv = test->priv;
496 struct cirrus_amp_cal_data result_data;
497 int i, ret;
498
499 cs_amp_lib_test_init_dummy_cal_blob(test, 8);
500
501 /* Make all the target values zero so they are ignored */
502 for (i = 0; i < priv->cal_blob->count; ++i) {
503 priv->cal_blob->data[i].calTarget[0] = 0;
504 priv->cal_blob->data[i].calTarget[1] = 0;
505 }
506
507 /* Redirect calls to get EFI data */
508 kunit_activate_static_stub(test,
509 cs_amp_test_hooks->get_efi_variable,
510 cs_amp_lib_test_get_efi_variable);
511
512 ret = cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev, 0, -1, &result_data);
513 KUNIT_EXPECT_EQ(test, ret, -ENOENT);
514
515 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
516 }
517
518 /*
519 * If an entry has a timestamp of 0 it should be ignored even if it has
520 * a matching target UID.
521 */
cs_amp_lib_test_get_efi_cal_empty_entry_test(struct kunit * test)522 static void cs_amp_lib_test_get_efi_cal_empty_entry_test(struct kunit *test)
523 {
524 struct cs_amp_lib_test_priv *priv = test->priv;
525 struct cirrus_amp_cal_data result_data;
526 u64 uid;
527
528 cs_amp_lib_test_init_dummy_cal_blob(test, 8);
529
530 /* Mark the 3rd entry invalid by zeroing calTime */
531 priv->cal_blob->data[2].calTime[0] = 0;
532 priv->cal_blob->data[2].calTime[1] = 0;
533
534 /* Get the UID value of the 3rd entry */
535 uid = priv->cal_blob->data[2].calTarget[1];
536 uid <<= 32;
537 uid |= priv->cal_blob->data[2].calTarget[0];
538
539 /* Redirect calls to get EFI data */
540 kunit_activate_static_stub(test,
541 cs_amp_test_hooks->get_efi_variable,
542 cs_amp_lib_test_get_efi_variable);
543
544 /* Lookup by UID should not find it */
545 KUNIT_EXPECT_EQ(test,
546 cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev,
547 uid, -1,
548 &result_data),
549 -ENOENT);
550
551 /* Get by index should ignore it */
552 KUNIT_EXPECT_EQ(test,
553 cs_amp_get_efi_calibration_data(&priv->amp_pdev.dev,
554 0, 2,
555 &result_data),
556 -ENOENT);
557
558 kunit_deactivate_static_stub(test, cs_amp_test_hooks->get_efi_variable);
559 }
560
561 static const struct cirrus_amp_cal_controls cs_amp_lib_test_calibration_controls = {
562 .alg_id = 0x9f210,
563 .mem_region = WMFW_ADSP2_YM,
564 .ambient = "CAL_AMBIENT",
565 .calr = "CAL_R",
566 .status = "CAL_STATUS",
567 .checksum = "CAL_CHECKSUM",
568 };
569
cs_amp_lib_test_write_cal_coeff(struct cs_dsp * dsp,const struct cirrus_amp_cal_controls * controls,const char * ctl_name,u32 val)570 static int cs_amp_lib_test_write_cal_coeff(struct cs_dsp *dsp,
571 const struct cirrus_amp_cal_controls *controls,
572 const char *ctl_name, u32 val)
573 {
574 struct kunit *test = kunit_get_current_test();
575 struct cs_amp_lib_test_priv *priv = test->priv;
576 struct cs_amp_lib_test_ctl_write_entry *entry;
577
578 KUNIT_ASSERT_NOT_ERR_OR_NULL(test, ctl_name);
579 KUNIT_EXPECT_PTR_EQ(test, controls, &cs_amp_lib_test_calibration_controls);
580
581 entry = kunit_kzalloc(test, sizeof(*entry), GFP_KERNEL);
582 KUNIT_ASSERT_NOT_ERR_OR_NULL(test, entry);
583
584 INIT_LIST_HEAD(&entry->list);
585 strscpy(entry->name, ctl_name, sizeof(entry->name));
586 entry->value = val;
587
588 list_add_tail(&entry->list, &priv->ctl_write_list);
589
590 return 0;
591 }
592
cs_amp_lib_test_write_cal_data_test(struct kunit * test)593 static void cs_amp_lib_test_write_cal_data_test(struct kunit *test)
594 {
595 struct cs_amp_lib_test_priv *priv = test->priv;
596 struct cs_amp_lib_test_ctl_write_entry *entry;
597 struct cirrus_amp_cal_data data;
598 struct cs_dsp *dsp;
599 int ret;
600
601 dsp = kunit_kzalloc(test, sizeof(*dsp), GFP_KERNEL);
602 KUNIT_ASSERT_NOT_ERR_OR_NULL(test, dsp);
603 dsp->dev = &priv->amp_pdev.dev;
604
605 get_random_bytes(&data, sizeof(data));
606
607 /* Redirect calls to write firmware controls */
608 kunit_activate_static_stub(test,
609 cs_amp_test_hooks->write_cal_coeff,
610 cs_amp_lib_test_write_cal_coeff);
611
612 ret = cs_amp_write_cal_coeffs(dsp, &cs_amp_lib_test_calibration_controls, &data);
613 KUNIT_EXPECT_EQ(test, ret, 0);
614
615 kunit_deactivate_static_stub(test, cs_amp_test_hooks->write_cal_coeff);
616
617 KUNIT_EXPECT_EQ(test, list_count_nodes(&priv->ctl_write_list), 4);
618
619 /* Checksum control must be written last */
620 entry = list_last_entry(&priv->ctl_write_list, typeof(*entry), list);
621 KUNIT_EXPECT_STREQ(test, entry->name, cs_amp_lib_test_calibration_controls.checksum);
622 KUNIT_EXPECT_EQ(test, entry->value, data.calR + 1);
623 list_del(&entry->list);
624
625 entry = list_first_entry(&priv->ctl_write_list, typeof(*entry), list);
626 KUNIT_EXPECT_STREQ(test, entry->name, cs_amp_lib_test_calibration_controls.ambient);
627 KUNIT_EXPECT_EQ(test, entry->value, data.calAmbient);
628 list_del(&entry->list);
629
630 entry = list_first_entry(&priv->ctl_write_list, typeof(*entry), list);
631 KUNIT_EXPECT_STREQ(test, entry->name, cs_amp_lib_test_calibration_controls.calr);
632 KUNIT_EXPECT_EQ(test, entry->value, data.calR);
633 list_del(&entry->list);
634
635 entry = list_first_entry(&priv->ctl_write_list, typeof(*entry), list);
636 KUNIT_EXPECT_STREQ(test, entry->name, cs_amp_lib_test_calibration_controls.status);
637 KUNIT_EXPECT_EQ(test, entry->value, data.calStatus);
638 }
639
cs_amp_lib_test_dev_release(struct device * dev)640 static void cs_amp_lib_test_dev_release(struct device *dev)
641 {
642 }
643
cs_amp_lib_test_case_init(struct kunit * test)644 static int cs_amp_lib_test_case_init(struct kunit *test)
645 {
646 struct cs_amp_lib_test_priv *priv;
647 int ret;
648
649 KUNIT_ASSERT_NOT_NULL(test, cs_amp_test_hooks);
650
651 priv = kunit_kzalloc(test, sizeof(*priv), GFP_KERNEL);
652 if (!priv)
653 return -ENOMEM;
654
655 test->priv = priv;
656 INIT_LIST_HEAD(&priv->ctl_write_list);
657
658 /* Create dummy amp driver dev */
659 priv->amp_pdev.name = "cs_amp_lib_test_drv";
660 priv->amp_pdev.id = -1;
661 priv->amp_pdev.dev.release = cs_amp_lib_test_dev_release;
662 ret = platform_device_register(&priv->amp_pdev);
663 KUNIT_ASSERT_GE_MSG(test, ret, 0, "Failed to register amp platform device\n");
664
665 return 0;
666 }
667
cs_amp_lib_test_case_exit(struct kunit * test)668 static void cs_amp_lib_test_case_exit(struct kunit *test)
669 {
670 struct cs_amp_lib_test_priv *priv = test->priv;
671
672 if (priv->amp_pdev.name)
673 platform_device_unregister(&priv->amp_pdev);
674 }
675
676 static const struct cs_amp_lib_test_param cs_amp_lib_test_get_cal_param_cases[] = {
677 { .num_amps = 2, .amp_index = 0 },
678 { .num_amps = 2, .amp_index = 1 },
679
680 { .num_amps = 3, .amp_index = 0 },
681 { .num_amps = 3, .amp_index = 1 },
682 { .num_amps = 3, .amp_index = 2 },
683
684 { .num_amps = 4, .amp_index = 0 },
685 { .num_amps = 4, .amp_index = 1 },
686 { .num_amps = 4, .amp_index = 2 },
687 { .num_amps = 4, .amp_index = 3 },
688
689 { .num_amps = 5, .amp_index = 0 },
690 { .num_amps = 5, .amp_index = 1 },
691 { .num_amps = 5, .amp_index = 2 },
692 { .num_amps = 5, .amp_index = 3 },
693 { .num_amps = 5, .amp_index = 4 },
694
695 { .num_amps = 6, .amp_index = 0 },
696 { .num_amps = 6, .amp_index = 1 },
697 { .num_amps = 6, .amp_index = 2 },
698 { .num_amps = 6, .amp_index = 3 },
699 { .num_amps = 6, .amp_index = 4 },
700 { .num_amps = 6, .amp_index = 5 },
701
702 { .num_amps = 8, .amp_index = 0 },
703 { .num_amps = 8, .amp_index = 1 },
704 { .num_amps = 8, .amp_index = 2 },
705 { .num_amps = 8, .amp_index = 3 },
706 { .num_amps = 8, .amp_index = 4 },
707 { .num_amps = 8, .amp_index = 5 },
708 { .num_amps = 8, .amp_index = 6 },
709 { .num_amps = 8, .amp_index = 7 },
710 };
711
cs_amp_lib_test_get_cal_param_desc(const struct cs_amp_lib_test_param * param,char * desc)712 static void cs_amp_lib_test_get_cal_param_desc(const struct cs_amp_lib_test_param *param,
713 char *desc)
714 {
715 snprintf(desc, KUNIT_PARAM_DESC_SIZE, "num_amps:%d amp_index:%d",
716 param->num_amps, param->amp_index);
717 }
718
719 KUNIT_ARRAY_PARAM(cs_amp_lib_test_get_cal, cs_amp_lib_test_get_cal_param_cases,
720 cs_amp_lib_test_get_cal_param_desc);
721
722 static struct kunit_case cs_amp_lib_test_cases[] = {
723 /* Tests for getting calibration data from EFI */
724 KUNIT_CASE(cs_amp_lib_test_cal_data_too_short_test),
725 KUNIT_CASE(cs_amp_lib_test_cal_count_too_big_test),
726 KUNIT_CASE(cs_amp_lib_test_no_cal_data_test),
727 KUNIT_CASE(cs_amp_lib_test_get_efi_cal_uid_not_found_noindex_test),
728 KUNIT_CASE(cs_amp_lib_test_get_efi_cal_uid_not_found_index_not_found_test),
729 KUNIT_CASE(cs_amp_lib_test_get_efi_cal_no_uid_index_not_found_test),
730 KUNIT_CASE(cs_amp_lib_test_get_efi_cal_no_uid_no_index_test),
731 KUNIT_CASE(cs_amp_lib_test_get_efi_cal_zero_not_matched_test),
732 KUNIT_CASE_PARAM(cs_amp_lib_test_get_efi_cal_by_uid_test,
733 cs_amp_lib_test_get_cal_gen_params),
734 KUNIT_CASE_PARAM(cs_amp_lib_test_get_efi_cal_by_index_unchecked_test,
735 cs_amp_lib_test_get_cal_gen_params),
736 KUNIT_CASE_PARAM(cs_amp_lib_test_get_efi_cal_by_index_checked_test,
737 cs_amp_lib_test_get_cal_gen_params),
738 KUNIT_CASE_PARAM(cs_amp_lib_test_get_efi_cal_by_index_uid_mismatch_test,
739 cs_amp_lib_test_get_cal_gen_params),
740 KUNIT_CASE_PARAM(cs_amp_lib_test_get_efi_cal_by_index_fallback_test,
741 cs_amp_lib_test_get_cal_gen_params),
742 KUNIT_CASE(cs_amp_lib_test_get_efi_cal_empty_entry_test),
743
744 /* Tests for writing calibration data */
745 KUNIT_CASE(cs_amp_lib_test_write_cal_data_test),
746
747 { } /* terminator */
748 };
749
750 static struct kunit_suite cs_amp_lib_test_suite = {
751 .name = "snd-soc-cs-amp-lib-test",
752 .init = cs_amp_lib_test_case_init,
753 .exit = cs_amp_lib_test_case_exit,
754 .test_cases = cs_amp_lib_test_cases,
755 };
756
757 kunit_test_suite(cs_amp_lib_test_suite);
758
759 MODULE_IMPORT_NS("SND_SOC_CS_AMP_LIB");
760 MODULE_DESCRIPTION("KUnit test for Cirrus Logic amplifier library");
761 MODULE_AUTHOR("Richard Fitzgerald <rf@opensource.cirrus.com>");
762 MODULE_LICENSE("GPL");
763