1 // SPDX-License-Identifier: GPL-2.0-only 2 /* 3 * soc-topology-test.c -- ALSA SoC Topology Kernel Unit Tests 4 * 5 * Copyright(c) 2021 Intel Corporation. 6 */ 7 8 #include <linux/firmware.h> 9 #include <sound/core.h> 10 #include <sound/soc.h> 11 #include <sound/soc-topology.h> 12 #include <kunit/device.h> 13 #include <kunit/test.h> 14 15 /* ===== HELPER FUNCTIONS =================================================== */ 16 17 /* 18 * snd_soc_component needs device to operate on (primarily for prints), create 19 * fake one, as we don't register with PCI or anything else 20 * device_driver name is used in some of the prints (fmt_single_name) so 21 * we also mock up minimal one 22 */ 23 static struct device *test_dev; 24 25 static int snd_soc_tplg_test_init(struct kunit *test) 26 { 27 test_dev = kunit_device_register(test, "sound-soc-topology-test"); 28 test_dev = get_device(test_dev); 29 if (!test_dev) 30 return -ENODEV; 31 32 return 0; 33 } 34 35 static void snd_soc_tplg_test_exit(struct kunit *test) 36 { 37 put_device(test_dev); 38 } 39 40 /* 41 * helper struct we use when registering component, as we load topology during 42 * component probe, we need to pass struct kunit somehow to probe function, so 43 * we can report test result 44 */ 45 struct kunit_soc_component { 46 struct kunit *kunit; 47 int expect; /* what result we expect when loading topology */ 48 struct snd_soc_component comp; 49 struct snd_soc_card card; 50 struct firmware fw; 51 }; 52 53 static int d_probe(struct snd_soc_component *component) 54 { 55 struct kunit_soc_component *kunit_comp = 56 container_of(component, struct kunit_soc_component, comp); 57 int ret; 58 59 ret = snd_soc_tplg_component_load(component, NULL, &kunit_comp->fw); 60 KUNIT_EXPECT_EQ_MSG(kunit_comp->kunit, kunit_comp->expect, ret, 61 "Failed topology load"); 62 63 return 0; 64 } 65 66 static void d_remove(struct snd_soc_component *component) 67 { 68 struct kunit_soc_component *kunit_comp = 69 container_of(component, struct kunit_soc_component, comp); 70 int ret; 71 72 ret = snd_soc_tplg_component_remove(component); 73 KUNIT_EXPECT_EQ(kunit_comp->kunit, 0, ret); 74 } 75 76 /* 77 * ASoC minimal boiler plate 78 */ 79 SND_SOC_DAILINK_DEF(dummy, DAILINK_COMP_ARRAY(COMP_DUMMY())); 80 81 SND_SOC_DAILINK_DEF(platform, DAILINK_COMP_ARRAY(COMP_PLATFORM("sound-soc-topology-test"))); 82 83 static struct snd_soc_dai_link kunit_dai_links[] = { 84 { 85 .name = "KUNIT Audio Port", 86 .id = 0, 87 .stream_name = "Audio Playback/Capture", 88 .nonatomic = 1, 89 .dynamic = 1, 90 .trigger = {SND_SOC_DPCM_TRIGGER_POST, SND_SOC_DPCM_TRIGGER_POST}, 91 .dpcm_playback = 1, 92 .dpcm_capture = 1, 93 SND_SOC_DAILINK_REG(dummy, dummy, platform), 94 }, 95 }; 96 97 static const struct snd_soc_component_driver test_component = { 98 .name = "sound-soc-topology-test", 99 .probe = d_probe, 100 .remove = d_remove, 101 }; 102 103 /* ===== TOPOLOGY TEMPLATES ================================================= */ 104 105 // Structural representation of topology which can be generated with: 106 // $ touch empty 107 // $ alsatplg -c empty -o empty.tplg 108 // $ xxd -i empty.tplg 109 110 struct tplg_tmpl_001 { 111 struct snd_soc_tplg_hdr header; 112 struct snd_soc_tplg_manifest manifest; 113 } __packed; 114 115 static struct tplg_tmpl_001 tplg_tmpl_empty = { 116 .header = { 117 .magic = cpu_to_le32(SND_SOC_TPLG_MAGIC), 118 .abi = cpu_to_le32(5), 119 .version = 0, 120 .type = cpu_to_le32(SND_SOC_TPLG_TYPE_MANIFEST), 121 .size = cpu_to_le32(sizeof(struct snd_soc_tplg_hdr)), 122 .vendor_type = 0, 123 .payload_size = cpu_to_le32(sizeof(struct snd_soc_tplg_manifest)), 124 .index = 0, 125 .count = cpu_to_le32(1), 126 }, 127 128 .manifest = { 129 .size = cpu_to_le32(sizeof(struct snd_soc_tplg_manifest)), 130 /* rest of fields is 0 */ 131 }, 132 }; 133 134 // Structural representation of topology containing SectionPCM 135 136 struct tplg_tmpl_002 { 137 struct snd_soc_tplg_hdr header; 138 struct snd_soc_tplg_manifest manifest; 139 struct snd_soc_tplg_hdr pcm_header; 140 struct snd_soc_tplg_pcm pcm; 141 } __packed; 142 143 static struct tplg_tmpl_002 tplg_tmpl_with_pcm = { 144 .header = { 145 .magic = cpu_to_le32(SND_SOC_TPLG_MAGIC), 146 .abi = cpu_to_le32(5), 147 .version = 0, 148 .type = cpu_to_le32(SND_SOC_TPLG_TYPE_MANIFEST), 149 .size = cpu_to_le32(sizeof(struct snd_soc_tplg_hdr)), 150 .vendor_type = 0, 151 .payload_size = cpu_to_le32(sizeof(struct snd_soc_tplg_manifest)), 152 .index = 0, 153 .count = cpu_to_le32(1), 154 }, 155 .manifest = { 156 .size = cpu_to_le32(sizeof(struct snd_soc_tplg_manifest)), 157 .pcm_elems = cpu_to_le32(1), 158 /* rest of fields is 0 */ 159 }, 160 .pcm_header = { 161 .magic = cpu_to_le32(SND_SOC_TPLG_MAGIC), 162 .abi = cpu_to_le32(5), 163 .version = 0, 164 .type = cpu_to_le32(SND_SOC_TPLG_TYPE_PCM), 165 .size = cpu_to_le32(sizeof(struct snd_soc_tplg_hdr)), 166 .vendor_type = 0, 167 .payload_size = cpu_to_le32(sizeof(struct snd_soc_tplg_pcm)), 168 .index = 0, 169 .count = cpu_to_le32(1), 170 }, 171 .pcm = { 172 .size = cpu_to_le32(sizeof(struct snd_soc_tplg_pcm)), 173 .pcm_name = "KUNIT Audio", 174 .dai_name = "kunit-audio-dai", 175 .pcm_id = 0, 176 .dai_id = 0, 177 .playback = cpu_to_le32(1), 178 .capture = cpu_to_le32(1), 179 .compress = 0, 180 .stream = { 181 [0] = { 182 .channels = cpu_to_le32(2), 183 }, 184 [1] = { 185 .channels = cpu_to_le32(2), 186 }, 187 }, 188 .num_streams = 0, 189 .caps = { 190 [0] = { 191 .name = "kunit-audio-playback", 192 .channels_min = cpu_to_le32(2), 193 .channels_max = cpu_to_le32(2), 194 }, 195 [1] = { 196 .name = "kunit-audio-capture", 197 .channels_min = cpu_to_le32(2), 198 .channels_max = cpu_to_le32(2), 199 }, 200 }, 201 .flag_mask = 0, 202 .flags = 0, 203 .priv = { 0 }, 204 }, 205 }; 206 207 /* ===== TEST CASES ========================================================= */ 208 209 // TEST CASE 210 // Test passing NULL component as parameter to snd_soc_tplg_component_load 211 212 /* 213 * need to override generic probe function with one using NULL when calling 214 * topology load during component initialization, we don't need .remove 215 * handler as load should fail 216 */ 217 static int d_probe_null_comp(struct snd_soc_component *component) 218 { 219 struct kunit_soc_component *kunit_comp = 220 container_of(component, struct kunit_soc_component, comp); 221 int ret; 222 223 /* instead of passing component pointer as first argument, pass NULL here */ 224 ret = snd_soc_tplg_component_load(NULL, NULL, &kunit_comp->fw); 225 KUNIT_EXPECT_EQ_MSG(kunit_comp->kunit, kunit_comp->expect, ret, 226 "Failed topology load"); 227 228 return 0; 229 } 230 231 static const struct snd_soc_component_driver test_component_null_comp = { 232 .name = "sound-soc-topology-test", 233 .probe = d_probe_null_comp, 234 }; 235 236 static void snd_soc_tplg_test_load_with_null_comp(struct kunit *test) 237 { 238 struct kunit_soc_component *kunit_comp; 239 int ret; 240 241 /* prepare */ 242 kunit_comp = kunit_kzalloc(test, sizeof(*kunit_comp), GFP_KERNEL); 243 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, kunit_comp); 244 kunit_comp->kunit = test; 245 kunit_comp->expect = -EINVAL; /* expect failure */ 246 247 kunit_comp->card.dev = test_dev, 248 kunit_comp->card.name = "kunit-card", 249 kunit_comp->card.owner = THIS_MODULE, 250 kunit_comp->card.dai_link = kunit_dai_links, 251 kunit_comp->card.num_links = ARRAY_SIZE(kunit_dai_links), 252 kunit_comp->card.fully_routed = true, 253 254 /* run test */ 255 ret = snd_soc_register_card(&kunit_comp->card); 256 if (ret != 0 && ret != -EPROBE_DEFER) 257 KUNIT_FAIL(test, "Failed to register card"); 258 259 ret = snd_soc_component_initialize(&kunit_comp->comp, &test_component_null_comp, test_dev); 260 KUNIT_EXPECT_EQ(test, 0, ret); 261 262 ret = snd_soc_add_component(&kunit_comp->comp, NULL, 0); 263 KUNIT_EXPECT_EQ(test, 0, ret); 264 265 /* cleanup */ 266 snd_soc_unregister_card(&kunit_comp->card); 267 snd_soc_unregister_component(test_dev); 268 } 269 270 // TEST CASE 271 // Test passing NULL ops as parameter to snd_soc_tplg_component_load 272 273 /* 274 * NULL ops is default case, we pass empty topology (fw), so we don't have 275 * anything to parse and just do nothing, which results in return 0; from 276 * calling soc_tplg_dapm_complete in soc_tplg_process_headers 277 */ 278 static void snd_soc_tplg_test_load_with_null_ops(struct kunit *test) 279 { 280 struct kunit_soc_component *kunit_comp; 281 int ret; 282 283 /* prepare */ 284 kunit_comp = kunit_kzalloc(test, sizeof(*kunit_comp), GFP_KERNEL); 285 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, kunit_comp); 286 kunit_comp->kunit = test; 287 kunit_comp->expect = 0; /* expect success */ 288 289 kunit_comp->card.dev = test_dev, 290 kunit_comp->card.name = "kunit-card", 291 kunit_comp->card.owner = THIS_MODULE, 292 kunit_comp->card.dai_link = kunit_dai_links, 293 kunit_comp->card.num_links = ARRAY_SIZE(kunit_dai_links), 294 kunit_comp->card.fully_routed = true, 295 296 /* run test */ 297 ret = snd_soc_register_card(&kunit_comp->card); 298 if (ret != 0 && ret != -EPROBE_DEFER) 299 KUNIT_FAIL(test, "Failed to register card"); 300 301 ret = snd_soc_component_initialize(&kunit_comp->comp, &test_component, test_dev); 302 KUNIT_EXPECT_EQ(test, 0, ret); 303 304 ret = snd_soc_add_component(&kunit_comp->comp, NULL, 0); 305 KUNIT_EXPECT_EQ(test, 0, ret); 306 307 /* cleanup */ 308 snd_soc_unregister_card(&kunit_comp->card); 309 310 snd_soc_unregister_component(test_dev); 311 } 312 313 // TEST CASE 314 // Test passing NULL fw as parameter to snd_soc_tplg_component_load 315 316 /* 317 * need to override generic probe function with one using NULL pointer to fw 318 * when calling topology load during component initialization, we don't need 319 * .remove handler as load should fail 320 */ 321 static int d_probe_null_fw(struct snd_soc_component *component) 322 { 323 struct kunit_soc_component *kunit_comp = 324 container_of(component, struct kunit_soc_component, comp); 325 int ret; 326 327 /* instead of passing fw pointer as third argument, pass NULL here */ 328 ret = snd_soc_tplg_component_load(component, NULL, NULL); 329 KUNIT_EXPECT_EQ_MSG(kunit_comp->kunit, kunit_comp->expect, ret, 330 "Failed topology load"); 331 332 return 0; 333 } 334 335 static const struct snd_soc_component_driver test_component_null_fw = { 336 .name = "sound-soc-topology-test", 337 .probe = d_probe_null_fw, 338 }; 339 340 static void snd_soc_tplg_test_load_with_null_fw(struct kunit *test) 341 { 342 struct kunit_soc_component *kunit_comp; 343 int ret; 344 345 /* prepare */ 346 kunit_comp = kunit_kzalloc(test, sizeof(*kunit_comp), GFP_KERNEL); 347 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, kunit_comp); 348 kunit_comp->kunit = test; 349 kunit_comp->expect = -EINVAL; /* expect failure */ 350 351 kunit_comp->card.dev = test_dev, 352 kunit_comp->card.name = "kunit-card", 353 kunit_comp->card.owner = THIS_MODULE, 354 kunit_comp->card.dai_link = kunit_dai_links, 355 kunit_comp->card.num_links = ARRAY_SIZE(kunit_dai_links), 356 kunit_comp->card.fully_routed = true, 357 358 /* run test */ 359 ret = snd_soc_register_card(&kunit_comp->card); 360 if (ret != 0 && ret != -EPROBE_DEFER) 361 KUNIT_FAIL(test, "Failed to register card"); 362 363 ret = snd_soc_component_initialize(&kunit_comp->comp, &test_component_null_fw, test_dev); 364 KUNIT_EXPECT_EQ(test, 0, ret); 365 366 ret = snd_soc_add_component(&kunit_comp->comp, NULL, 0); 367 KUNIT_EXPECT_EQ(test, 0, ret); 368 369 /* cleanup */ 370 snd_soc_unregister_card(&kunit_comp->card); 371 372 snd_soc_unregister_component(test_dev); 373 } 374 375 // TEST CASE 376 // Test passing "empty" topology file 377 static void snd_soc_tplg_test_load_empty_tplg(struct kunit *test) 378 { 379 struct kunit_soc_component *kunit_comp; 380 struct tplg_tmpl_001 *data; 381 int size; 382 int ret; 383 384 /* prepare */ 385 kunit_comp = kunit_kzalloc(test, sizeof(*kunit_comp), GFP_KERNEL); 386 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, kunit_comp); 387 kunit_comp->kunit = test; 388 kunit_comp->expect = 0; /* expect success */ 389 390 size = sizeof(tplg_tmpl_empty); 391 data = kunit_kzalloc(kunit_comp->kunit, size, GFP_KERNEL); 392 KUNIT_EXPECT_NOT_ERR_OR_NULL(kunit_comp->kunit, data); 393 394 memcpy(data, &tplg_tmpl_empty, sizeof(tplg_tmpl_empty)); 395 396 kunit_comp->fw.data = (u8 *)data; 397 kunit_comp->fw.size = size; 398 399 kunit_comp->card.dev = test_dev, 400 kunit_comp->card.name = "kunit-card", 401 kunit_comp->card.owner = THIS_MODULE, 402 kunit_comp->card.dai_link = kunit_dai_links, 403 kunit_comp->card.num_links = ARRAY_SIZE(kunit_dai_links), 404 kunit_comp->card.fully_routed = true, 405 406 /* run test */ 407 ret = snd_soc_register_card(&kunit_comp->card); 408 if (ret != 0 && ret != -EPROBE_DEFER) 409 KUNIT_FAIL(test, "Failed to register card"); 410 411 ret = snd_soc_component_initialize(&kunit_comp->comp, &test_component, test_dev); 412 KUNIT_EXPECT_EQ(test, 0, ret); 413 414 ret = snd_soc_add_component(&kunit_comp->comp, NULL, 0); 415 KUNIT_EXPECT_EQ(test, 0, ret); 416 417 /* cleanup */ 418 snd_soc_unregister_card(&kunit_comp->card); 419 420 snd_soc_unregister_component(test_dev); 421 } 422 423 // TEST CASE 424 // Test "empty" topology file, but with bad "magic" 425 // In theory we could loop through all possible bad values, but it takes too 426 // long, so just use SND_SOC_TPLG_MAGIC + 1 427 static void snd_soc_tplg_test_load_empty_tplg_bad_magic(struct kunit *test) 428 { 429 struct kunit_soc_component *kunit_comp; 430 struct tplg_tmpl_001 *data; 431 int size; 432 int ret; 433 434 /* prepare */ 435 kunit_comp = kunit_kzalloc(test, sizeof(*kunit_comp), GFP_KERNEL); 436 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, kunit_comp); 437 kunit_comp->kunit = test; 438 kunit_comp->expect = -EINVAL; /* expect failure */ 439 440 size = sizeof(tplg_tmpl_empty); 441 data = kunit_kzalloc(kunit_comp->kunit, size, GFP_KERNEL); 442 KUNIT_EXPECT_NOT_ERR_OR_NULL(kunit_comp->kunit, data); 443 444 memcpy(data, &tplg_tmpl_empty, sizeof(tplg_tmpl_empty)); 445 /* 446 * override abi 447 * any value != magic number is wrong 448 */ 449 data->header.magic = cpu_to_le32(SND_SOC_TPLG_MAGIC + 1); 450 451 kunit_comp->fw.data = (u8 *)data; 452 kunit_comp->fw.size = size; 453 454 kunit_comp->card.dev = test_dev, 455 kunit_comp->card.name = "kunit-card", 456 kunit_comp->card.owner = THIS_MODULE, 457 kunit_comp->card.dai_link = kunit_dai_links, 458 kunit_comp->card.num_links = ARRAY_SIZE(kunit_dai_links), 459 kunit_comp->card.fully_routed = true, 460 461 /* run test */ 462 ret = snd_soc_register_card(&kunit_comp->card); 463 if (ret != 0 && ret != -EPROBE_DEFER) 464 KUNIT_FAIL(test, "Failed to register card"); 465 466 ret = snd_soc_component_initialize(&kunit_comp->comp, &test_component, test_dev); 467 KUNIT_EXPECT_EQ(test, 0, ret); 468 469 ret = snd_soc_add_component(&kunit_comp->comp, NULL, 0); 470 KUNIT_EXPECT_EQ(test, 0, ret); 471 472 /* cleanup */ 473 snd_soc_unregister_card(&kunit_comp->card); 474 475 snd_soc_unregister_component(test_dev); 476 } 477 478 // TEST CASE 479 // Test "empty" topology file, but with bad "abi" 480 // In theory we could loop through all possible bad values, but it takes too 481 // long, so just use SND_SOC_TPLG_ABI_VERSION + 1 482 static void snd_soc_tplg_test_load_empty_tplg_bad_abi(struct kunit *test) 483 { 484 struct kunit_soc_component *kunit_comp; 485 struct tplg_tmpl_001 *data; 486 int size; 487 int ret; 488 489 /* prepare */ 490 kunit_comp = kunit_kzalloc(test, sizeof(*kunit_comp), GFP_KERNEL); 491 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, kunit_comp); 492 kunit_comp->kunit = test; 493 kunit_comp->expect = -EINVAL; /* expect failure */ 494 495 size = sizeof(tplg_tmpl_empty); 496 data = kunit_kzalloc(kunit_comp->kunit, size, GFP_KERNEL); 497 KUNIT_EXPECT_NOT_ERR_OR_NULL(kunit_comp->kunit, data); 498 499 memcpy(data, &tplg_tmpl_empty, sizeof(tplg_tmpl_empty)); 500 /* 501 * override abi 502 * any value != accepted range is wrong 503 */ 504 data->header.abi = cpu_to_le32(SND_SOC_TPLG_ABI_VERSION + 1); 505 506 kunit_comp->fw.data = (u8 *)data; 507 kunit_comp->fw.size = size; 508 509 kunit_comp->card.dev = test_dev, 510 kunit_comp->card.name = "kunit-card", 511 kunit_comp->card.owner = THIS_MODULE, 512 kunit_comp->card.dai_link = kunit_dai_links, 513 kunit_comp->card.num_links = ARRAY_SIZE(kunit_dai_links), 514 kunit_comp->card.fully_routed = true, 515 516 /* run test */ 517 ret = snd_soc_register_card(&kunit_comp->card); 518 if (ret != 0 && ret != -EPROBE_DEFER) 519 KUNIT_FAIL(test, "Failed to register card"); 520 521 ret = snd_soc_component_initialize(&kunit_comp->comp, &test_component, test_dev); 522 KUNIT_EXPECT_EQ(test, 0, ret); 523 524 ret = snd_soc_add_component(&kunit_comp->comp, NULL, 0); 525 KUNIT_EXPECT_EQ(test, 0, ret); 526 527 /* cleanup */ 528 snd_soc_unregister_card(&kunit_comp->card); 529 530 snd_soc_unregister_component(test_dev); 531 } 532 533 // TEST CASE 534 // Test "empty" topology file, but with bad "size" 535 // In theory we could loop through all possible bad values, but it takes too 536 // long, so just use sizeof(struct snd_soc_tplg_hdr) + 1 537 static void snd_soc_tplg_test_load_empty_tplg_bad_size(struct kunit *test) 538 { 539 struct kunit_soc_component *kunit_comp; 540 struct tplg_tmpl_001 *data; 541 int size; 542 int ret; 543 544 /* prepare */ 545 kunit_comp = kunit_kzalloc(test, sizeof(*kunit_comp), GFP_KERNEL); 546 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, kunit_comp); 547 kunit_comp->kunit = test; 548 kunit_comp->expect = -EINVAL; /* expect failure */ 549 550 size = sizeof(tplg_tmpl_empty); 551 data = kunit_kzalloc(kunit_comp->kunit, size, GFP_KERNEL); 552 KUNIT_EXPECT_NOT_ERR_OR_NULL(kunit_comp->kunit, data); 553 554 memcpy(data, &tplg_tmpl_empty, sizeof(tplg_tmpl_empty)); 555 /* 556 * override size 557 * any value != struct size is wrong 558 */ 559 data->header.size = cpu_to_le32(sizeof(struct snd_soc_tplg_hdr) + 1); 560 561 kunit_comp->fw.data = (u8 *)data; 562 kunit_comp->fw.size = size; 563 564 kunit_comp->card.dev = test_dev, 565 kunit_comp->card.name = "kunit-card", 566 kunit_comp->card.owner = THIS_MODULE, 567 kunit_comp->card.dai_link = kunit_dai_links, 568 kunit_comp->card.num_links = ARRAY_SIZE(kunit_dai_links), 569 kunit_comp->card.fully_routed = true, 570 571 /* run test */ 572 ret = snd_soc_register_card(&kunit_comp->card); 573 if (ret != 0 && ret != -EPROBE_DEFER) 574 KUNIT_FAIL(test, "Failed to register card"); 575 576 ret = snd_soc_component_initialize(&kunit_comp->comp, &test_component, test_dev); 577 KUNIT_EXPECT_EQ(test, 0, ret); 578 579 ret = snd_soc_add_component(&kunit_comp->comp, NULL, 0); 580 KUNIT_EXPECT_EQ(test, 0, ret); 581 582 /* cleanup */ 583 snd_soc_unregister_card(&kunit_comp->card); 584 585 snd_soc_unregister_component(test_dev); 586 } 587 588 // TEST CASE 589 // Test "empty" topology file, but with bad "payload_size" 590 // In theory we could loop through all possible bad values, but it takes too 591 // long, so just use the known wrong one 592 static void snd_soc_tplg_test_load_empty_tplg_bad_payload_size(struct kunit *test) 593 { 594 struct kunit_soc_component *kunit_comp; 595 struct tplg_tmpl_001 *data; 596 int size; 597 int ret; 598 599 /* prepare */ 600 kunit_comp = kunit_kzalloc(test, sizeof(*kunit_comp), GFP_KERNEL); 601 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, kunit_comp); 602 kunit_comp->kunit = test; 603 kunit_comp->expect = -EINVAL; /* expect failure */ 604 605 size = sizeof(tplg_tmpl_empty); 606 data = kunit_kzalloc(kunit_comp->kunit, size, GFP_KERNEL); 607 KUNIT_EXPECT_NOT_ERR_OR_NULL(kunit_comp->kunit, data); 608 609 memcpy(data, &tplg_tmpl_empty, sizeof(tplg_tmpl_empty)); 610 /* 611 * override payload size 612 * there is only explicit check for 0, so check with it, other values 613 * are handled by just not reading behind EOF 614 */ 615 data->header.payload_size = 0; 616 617 kunit_comp->fw.data = (u8 *)data; 618 kunit_comp->fw.size = size; 619 620 kunit_comp->card.dev = test_dev, 621 kunit_comp->card.name = "kunit-card", 622 kunit_comp->card.owner = THIS_MODULE, 623 kunit_comp->card.dai_link = kunit_dai_links, 624 kunit_comp->card.num_links = ARRAY_SIZE(kunit_dai_links), 625 kunit_comp->card.fully_routed = true, 626 627 /* run test */ 628 ret = snd_soc_register_card(&kunit_comp->card); 629 if (ret != 0 && ret != -EPROBE_DEFER) 630 KUNIT_FAIL(test, "Failed to register card"); 631 632 ret = snd_soc_component_initialize(&kunit_comp->comp, &test_component, test_dev); 633 KUNIT_EXPECT_EQ(test, 0, ret); 634 635 ret = snd_soc_add_component(&kunit_comp->comp, NULL, 0); 636 KUNIT_EXPECT_EQ(test, 0, ret); 637 638 /* cleanup */ 639 snd_soc_unregister_component(test_dev); 640 641 snd_soc_unregister_card(&kunit_comp->card); 642 } 643 644 // TEST CASE 645 // Test passing topology file with PCM definition 646 static void snd_soc_tplg_test_load_pcm_tplg(struct kunit *test) 647 { 648 struct kunit_soc_component *kunit_comp; 649 u8 *data; 650 int size; 651 int ret; 652 653 /* prepare */ 654 kunit_comp = kunit_kzalloc(test, sizeof(*kunit_comp), GFP_KERNEL); 655 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, kunit_comp); 656 kunit_comp->kunit = test; 657 kunit_comp->expect = 0; /* expect success */ 658 659 size = sizeof(tplg_tmpl_with_pcm); 660 data = kunit_kzalloc(kunit_comp->kunit, size, GFP_KERNEL); 661 KUNIT_EXPECT_NOT_ERR_OR_NULL(kunit_comp->kunit, data); 662 663 memcpy(data, &tplg_tmpl_with_pcm, sizeof(tplg_tmpl_with_pcm)); 664 665 kunit_comp->fw.data = data; 666 kunit_comp->fw.size = size; 667 668 kunit_comp->card.dev = test_dev, 669 kunit_comp->card.name = "kunit-card", 670 kunit_comp->card.owner = THIS_MODULE, 671 kunit_comp->card.dai_link = kunit_dai_links, 672 kunit_comp->card.num_links = ARRAY_SIZE(kunit_dai_links), 673 kunit_comp->card.fully_routed = true, 674 675 /* run test */ 676 ret = snd_soc_register_card(&kunit_comp->card); 677 if (ret != 0 && ret != -EPROBE_DEFER) 678 KUNIT_FAIL(test, "Failed to register card"); 679 680 ret = snd_soc_component_initialize(&kunit_comp->comp, &test_component, test_dev); 681 KUNIT_EXPECT_EQ(test, 0, ret); 682 683 ret = snd_soc_add_component(&kunit_comp->comp, NULL, 0); 684 KUNIT_EXPECT_EQ(test, 0, ret); 685 686 snd_soc_unregister_component(test_dev); 687 688 /* cleanup */ 689 snd_soc_unregister_card(&kunit_comp->card); 690 } 691 692 // TEST CASE 693 // Test passing topology file with PCM definition 694 // with component reload 695 static void snd_soc_tplg_test_load_pcm_tplg_reload_comp(struct kunit *test) 696 { 697 struct kunit_soc_component *kunit_comp; 698 u8 *data; 699 int size; 700 int ret; 701 int i; 702 703 /* prepare */ 704 kunit_comp = kunit_kzalloc(test, sizeof(*kunit_comp), GFP_KERNEL); 705 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, kunit_comp); 706 kunit_comp->kunit = test; 707 kunit_comp->expect = 0; /* expect success */ 708 709 size = sizeof(tplg_tmpl_with_pcm); 710 data = kunit_kzalloc(kunit_comp->kunit, size, GFP_KERNEL); 711 KUNIT_EXPECT_NOT_ERR_OR_NULL(kunit_comp->kunit, data); 712 713 memcpy(data, &tplg_tmpl_with_pcm, sizeof(tplg_tmpl_with_pcm)); 714 715 kunit_comp->fw.data = data; 716 kunit_comp->fw.size = size; 717 718 kunit_comp->card.dev = test_dev, 719 kunit_comp->card.name = "kunit-card", 720 kunit_comp->card.owner = THIS_MODULE, 721 kunit_comp->card.dai_link = kunit_dai_links, 722 kunit_comp->card.num_links = ARRAY_SIZE(kunit_dai_links), 723 kunit_comp->card.fully_routed = true, 724 725 /* run test */ 726 ret = snd_soc_register_card(&kunit_comp->card); 727 if (ret != 0 && ret != -EPROBE_DEFER) 728 KUNIT_FAIL(test, "Failed to register card"); 729 730 for (i = 0; i < 100; i++) { 731 ret = snd_soc_component_initialize(&kunit_comp->comp, &test_component, test_dev); 732 KUNIT_EXPECT_EQ(test, 0, ret); 733 734 ret = snd_soc_add_component(&kunit_comp->comp, NULL, 0); 735 KUNIT_EXPECT_EQ(test, 0, ret); 736 737 snd_soc_unregister_component(test_dev); 738 } 739 740 /* cleanup */ 741 snd_soc_unregister_card(&kunit_comp->card); 742 } 743 744 // TEST CASE 745 // Test passing topology file with PCM definition 746 // with card reload 747 static void snd_soc_tplg_test_load_pcm_tplg_reload_card(struct kunit *test) 748 { 749 struct kunit_soc_component *kunit_comp; 750 u8 *data; 751 int size; 752 int ret; 753 int i; 754 755 /* prepare */ 756 kunit_comp = kunit_kzalloc(test, sizeof(*kunit_comp), GFP_KERNEL); 757 KUNIT_EXPECT_NOT_ERR_OR_NULL(test, kunit_comp); 758 kunit_comp->kunit = test; 759 kunit_comp->expect = 0; /* expect success */ 760 761 size = sizeof(tplg_tmpl_with_pcm); 762 data = kunit_kzalloc(kunit_comp->kunit, size, GFP_KERNEL); 763 KUNIT_EXPECT_NOT_ERR_OR_NULL(kunit_comp->kunit, data); 764 765 memcpy(data, &tplg_tmpl_with_pcm, sizeof(tplg_tmpl_with_pcm)); 766 767 kunit_comp->fw.data = data; 768 kunit_comp->fw.size = size; 769 770 kunit_comp->card.dev = test_dev, 771 kunit_comp->card.name = "kunit-card", 772 kunit_comp->card.owner = THIS_MODULE, 773 kunit_comp->card.dai_link = kunit_dai_links, 774 kunit_comp->card.num_links = ARRAY_SIZE(kunit_dai_links), 775 kunit_comp->card.fully_routed = true, 776 777 /* run test */ 778 ret = snd_soc_component_initialize(&kunit_comp->comp, &test_component, test_dev); 779 KUNIT_EXPECT_EQ(test, 0, ret); 780 781 ret = snd_soc_add_component(&kunit_comp->comp, NULL, 0); 782 KUNIT_EXPECT_EQ(test, 0, ret); 783 784 for (i = 0; i < 100; i++) { 785 ret = snd_soc_register_card(&kunit_comp->card); 786 if (ret != 0 && ret != -EPROBE_DEFER) 787 KUNIT_FAIL(test, "Failed to register card"); 788 789 snd_soc_unregister_card(&kunit_comp->card); 790 } 791 792 /* cleanup */ 793 snd_soc_unregister_component(test_dev); 794 } 795 796 /* ===== KUNIT MODULE DEFINITIONS =========================================== */ 797 798 static struct kunit_case snd_soc_tplg_test_cases[] = { 799 KUNIT_CASE(snd_soc_tplg_test_load_with_null_comp), 800 KUNIT_CASE(snd_soc_tplg_test_load_with_null_ops), 801 KUNIT_CASE(snd_soc_tplg_test_load_with_null_fw), 802 KUNIT_CASE(snd_soc_tplg_test_load_empty_tplg), 803 KUNIT_CASE(snd_soc_tplg_test_load_empty_tplg_bad_magic), 804 KUNIT_CASE(snd_soc_tplg_test_load_empty_tplg_bad_abi), 805 KUNIT_CASE(snd_soc_tplg_test_load_empty_tplg_bad_size), 806 KUNIT_CASE(snd_soc_tplg_test_load_empty_tplg_bad_payload_size), 807 KUNIT_CASE(snd_soc_tplg_test_load_pcm_tplg), 808 KUNIT_CASE(snd_soc_tplg_test_load_pcm_tplg_reload_comp), 809 KUNIT_CASE(snd_soc_tplg_test_load_pcm_tplg_reload_card), 810 {} 811 }; 812 813 static struct kunit_suite snd_soc_tplg_test_suite = { 814 .name = "snd_soc_tplg_test", 815 .init = snd_soc_tplg_test_init, 816 .exit = snd_soc_tplg_test_exit, 817 .test_cases = snd_soc_tplg_test_cases, 818 }; 819 820 kunit_test_suites(&snd_soc_tplg_test_suite); 821 822 MODULE_LICENSE("GPL"); 823