18b98745dSDavid Hildenbrand /* SPDX-License-Identifier: GPL-2.0-only */
28b98745dSDavid Hildenbrand /*
38b98745dSDavid Hildenbrand * Floating interrupt tests.
48b98745dSDavid Hildenbrand *
58b98745dSDavid Hildenbrand * Copyright 2021 Red Hat Inc
68b98745dSDavid Hildenbrand *
78b98745dSDavid Hildenbrand * Authors:
88b98745dSDavid Hildenbrand * David Hildenbrand <david@redhat.com>
98b98745dSDavid Hildenbrand */
108b98745dSDavid Hildenbrand #include <libcflat.h>
118b98745dSDavid Hildenbrand #include <asm/asm-offsets.h>
128b98745dSDavid Hildenbrand #include <asm/interrupt.h>
138b98745dSDavid Hildenbrand #include <asm/page.h>
148b98745dSDavid Hildenbrand #include <asm-generic/barrier.h>
158b98745dSDavid Hildenbrand
168b98745dSDavid Hildenbrand #include <sclp.h>
178b98745dSDavid Hildenbrand #include <smp.h>
188b98745dSDavid Hildenbrand #include <alloc_page.h>
198b98745dSDavid Hildenbrand
wait_for_sclp_int(void)208b98745dSDavid Hildenbrand static void wait_for_sclp_int(void)
218b98745dSDavid Hildenbrand {
228b98745dSDavid Hildenbrand /* Enable SCLP interrupts on this CPU only. */
238b98745dSDavid Hildenbrand ctl_set_bit(0, CTL0_SERVICE_SIGNAL);
248b98745dSDavid Hildenbrand
258b98745dSDavid Hildenbrand /* Enable external interrupts and go to the wait state. */
268b98745dSDavid Hildenbrand wait_for_interrupt(PSW_MASK_EXT);
278b98745dSDavid Hildenbrand }
288b98745dSDavid Hildenbrand
298b98745dSDavid Hildenbrand /*
308b98745dSDavid Hildenbrand * Some KVM versions might mix CPUs when looking for a floating IRQ target,
318b98745dSDavid Hildenbrand * accidentially detecting a stopped CPU as waiting and resulting in the actually
328b98745dSDavid Hildenbrand * waiting CPU not getting woken up for the interrupt.
338b98745dSDavid Hildenbrand */
test_wait_state_delivery(void)348b98745dSDavid Hildenbrand static void test_wait_state_delivery(void)
358b98745dSDavid Hildenbrand {
368b98745dSDavid Hildenbrand SCCBHeader *h;
378b98745dSDavid Hildenbrand int ret;
388b98745dSDavid Hildenbrand
398b98745dSDavid Hildenbrand report_prefix_push("wait state delivery");
408b98745dSDavid Hildenbrand
418b98745dSDavid Hildenbrand if (smp_query_num_cpus() < 3) {
428b98745dSDavid Hildenbrand report_skip("need at least 3 CPUs for this test");
438b98745dSDavid Hildenbrand goto out;
448b98745dSDavid Hildenbrand }
458b98745dSDavid Hildenbrand
465a2e2958SClaudio Imbrenda /* Stop CPU #2. It must succeed because we have at least 3 CPUs */
478b98745dSDavid Hildenbrand ret = smp_cpu_stop(2);
485a2e2958SClaudio Imbrenda assert(!ret);
498b98745dSDavid Hildenbrand
508b98745dSDavid Hildenbrand /*
515a2e2958SClaudio Imbrenda * We're going to perform an SCLP service call but expect the
525a2e2958SClaudio Imbrenda * interrupt on CPU #1 while it is in the wait state.
538b98745dSDavid Hildenbrand */
548b98745dSDavid Hildenbrand sclp_mark_busy();
558b98745dSDavid Hildenbrand
568b98745dSDavid Hildenbrand /* Start CPU #1 and let it wait for the interrupt. */
57*f514b4f7SClaudio Imbrenda ret = smp_cpu_setup(1, PSW_WITH_CUR_MASK(wait_for_sclp_int));
585a2e2958SClaudio Imbrenda /* This must not fail because we have at least 3 CPUs */
595a2e2958SClaudio Imbrenda assert(!ret);
608b98745dSDavid Hildenbrand
618b98745dSDavid Hildenbrand /*
628b98745dSDavid Hildenbrand * We'd have to jump trough some hoops to sense e.g., via SIGP
638b98745dSDavid Hildenbrand * CONDITIONAL EMERGENCY SIGNAL if CPU #1 is already in the
648b98745dSDavid Hildenbrand * wait state.
658b98745dSDavid Hildenbrand *
668b98745dSDavid Hildenbrand * Although not completely reliable, use SIGP SENSE RUNNING STATUS
678b98745dSDavid Hildenbrand * until not reported as running -- after all, our SCLP processing
688b98745dSDavid Hildenbrand * will take some time as well and smp_cpu_setup() returns when we're
698b98745dSDavid Hildenbrand * either already in wait_for_sclp_int() or just about to execute it.
708b98745dSDavid Hildenbrand */
718b98745dSDavid Hildenbrand while(smp_sense_running_status(1));
728b98745dSDavid Hildenbrand
730f26dca4SJanosch Frank h = alloc_pages_flags(0, AREA_DMA31);
748b98745dSDavid Hildenbrand h->length = 4096;
758b98745dSDavid Hildenbrand ret = servc(SCLP_CMDW_READ_CPU_INFO, __pa(h));
768b98745dSDavid Hildenbrand if (ret) {
778b98745dSDavid Hildenbrand sclp_clear_busy();
788b98745dSDavid Hildenbrand report_fail("SCLP_CMDW_READ_CPU_INFO failed");
798b98745dSDavid Hildenbrand goto out_destroy;
808b98745dSDavid Hildenbrand }
818b98745dSDavid Hildenbrand
828b98745dSDavid Hildenbrand /*
838b98745dSDavid Hildenbrand * Wait until the interrupt gets delivered on CPU #1, marking the
848b98745dSDavid Hildenbrand * SCLP requests as done.
858b98745dSDavid Hildenbrand */
868b98745dSDavid Hildenbrand sclp_wait_busy();
878b98745dSDavid Hildenbrand
888b98745dSDavid Hildenbrand report(true, "sclp interrupt delivered");
898b98745dSDavid Hildenbrand
908b98745dSDavid Hildenbrand out_destroy:
918b98745dSDavid Hildenbrand free_page(h);
928b98745dSDavid Hildenbrand smp_cpu_destroy(1);
938b98745dSDavid Hildenbrand out:
948b98745dSDavid Hildenbrand report_prefix_pop();
958b98745dSDavid Hildenbrand }
968b98745dSDavid Hildenbrand
main(void)978b98745dSDavid Hildenbrand int main(void)
988b98745dSDavid Hildenbrand {
998b98745dSDavid Hildenbrand report_prefix_push("firq");
1008b98745dSDavid Hildenbrand
1018b98745dSDavid Hildenbrand test_wait_state_delivery();
1028b98745dSDavid Hildenbrand
1038b98745dSDavid Hildenbrand report_prefix_pop();
1048b98745dSDavid Hildenbrand return report_summary();
1058b98745dSDavid Hildenbrand }
106