]> Git Repo - linux.git/blob - sound/soc/codecs/cs-amp-lib-test.c
Linux 6.14-rc3
[linux.git] / sound / soc / codecs / cs-amp-lib-test.c
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
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
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 */
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 */
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 */
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 */
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 */
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 */
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 */
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. */
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. */
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. */
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  */
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  */
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  */
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  */
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  */
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. */
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  */
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  */
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
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
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
640 static void cs_amp_lib_test_dev_release(struct device *dev)
641 {
642 }
643
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
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
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 <[email protected]>");
762 MODULE_LICENSE("GPL");
This page took 0.073654 seconds and 4 git commands to generate.