xref: /kvm-unit-tests/lib/powerpc/smp.c (revision 8b10d4fadc42e83fabc47fbdbe66d83a1e8f1438)
1d129d8a1SSuraj Jitindar Singh /*
2d129d8a1SSuraj Jitindar Singh  * Secondary cpu support
3d129d8a1SSuraj Jitindar Singh  *
4d129d8a1SSuraj Jitindar Singh  * Copyright 2016 Suraj Jitindar Singh, IBM.
5d129d8a1SSuraj Jitindar Singh  *
6d129d8a1SSuraj Jitindar Singh  * This work is licensed under the terms of the GNU LGPL, version 2.
7d129d8a1SSuraj Jitindar Singh  */
8d129d8a1SSuraj Jitindar Singh 
9d129d8a1SSuraj Jitindar Singh #include <devicetree.h>
10*8b10d4faSNicholas Piggin #include <asm/time.h>
11d129d8a1SSuraj Jitindar Singh #include <asm/setup.h>
12d129d8a1SSuraj Jitindar Singh #include <asm/rtas.h>
13d129d8a1SSuraj Jitindar Singh #include <asm/smp.h>
14d129d8a1SSuraj Jitindar Singh 
15d129d8a1SSuraj Jitindar Singh int nr_threads;
16d129d8a1SSuraj Jitindar Singh 
17d129d8a1SSuraj Jitindar Singh struct secondary_entry_data {
18d129d8a1SSuraj Jitindar Singh 	secondary_entry_fn entry;
19d129d8a1SSuraj Jitindar Singh 	uint64_t r3;
20d129d8a1SSuraj Jitindar Singh 	int nr_started;
21d129d8a1SSuraj Jitindar Singh };
22d129d8a1SSuraj Jitindar Singh 
23d129d8a1SSuraj Jitindar Singh /*
24d129d8a1SSuraj Jitindar Singh  * Start stopped thread cpu_id at entry
25d129d8a1SSuraj Jitindar Singh  * Returns:	<0 on failure to start stopped cpu
26d129d8a1SSuraj Jitindar Singh  *		0  on success
27d129d8a1SSuraj Jitindar Singh  *		>0 on cpu not in stopped state
28d129d8a1SSuraj Jitindar Singh  */
29d129d8a1SSuraj Jitindar Singh int start_thread(int cpu_id, secondary_entry_fn entry, uint32_t r3)
30d129d8a1SSuraj Jitindar Singh {
312565dce1SThomas Huth 	uint32_t query_token, start_token;
322565dce1SThomas Huth 	int outputs[1], ret;
33d129d8a1SSuraj Jitindar Singh 
342565dce1SThomas Huth 	ret = rtas_token("query-cpu-stopped-state", &query_token);
352565dce1SThomas Huth 	assert(ret == 0);
362565dce1SThomas Huth 	ret = rtas_token("start-cpu", &start_token);
372565dce1SThomas Huth 	assert(ret == 0);
38d129d8a1SSuraj Jitindar Singh 
39d129d8a1SSuraj Jitindar Singh 	ret = rtas_call(query_token, 1, 2, outputs, cpu_id);
40d129d8a1SSuraj Jitindar Singh 	if (ret) {
41d129d8a1SSuraj Jitindar Singh 		printf("query-cpu-stopped-state failed for cpu %d\n", cpu_id);
42d129d8a1SSuraj Jitindar Singh 	} else if (!outputs[0]) { /* cpu in stopped state */
43d129d8a1SSuraj Jitindar Singh 		ret = rtas_call(start_token, 3, 1, NULL, cpu_id, entry, r3);
44d129d8a1SSuraj Jitindar Singh 		if (ret)
45d129d8a1SSuraj Jitindar Singh 			printf("failed to start cpu %d\n", cpu_id);
46d129d8a1SSuraj Jitindar Singh 	} else { /* cpu not in stopped state */
47d129d8a1SSuraj Jitindar Singh 		ret = outputs[0];
48d129d8a1SSuraj Jitindar Singh 	}
49d129d8a1SSuraj Jitindar Singh 
50d129d8a1SSuraj Jitindar Singh 	return ret;
51d129d8a1SSuraj Jitindar Singh }
52d129d8a1SSuraj Jitindar Singh 
53d129d8a1SSuraj Jitindar Singh /*
54d129d8a1SSuraj Jitindar Singh  * Start all stopped threads (vcpus) on cpu_node
55d129d8a1SSuraj Jitindar Singh  * Returns: Number of stopped cpus which were successfully started
56d129d8a1SSuraj Jitindar Singh  */
57d129d8a1SSuraj Jitindar Singh struct start_threads start_cpu(int cpu_node, secondary_entry_fn entry,
58d129d8a1SSuraj Jitindar Singh 			       uint32_t r3)
59d129d8a1SSuraj Jitindar Singh {
60d129d8a1SSuraj Jitindar Singh 	int len, i, nr_threads, nr_started = 0;
61d129d8a1SSuraj Jitindar Singh 	const struct fdt_property *prop;
62d129d8a1SSuraj Jitindar Singh 	u32 *threads;
63d129d8a1SSuraj Jitindar Singh 
64d129d8a1SSuraj Jitindar Singh 	/* Get the id array of threads on this cpu_node */
65d129d8a1SSuraj Jitindar Singh 	prop = fdt_get_property(dt_fdt(), cpu_node,
66d129d8a1SSuraj Jitindar Singh 				"ibm,ppc-interrupt-server#s", &len);
67d129d8a1SSuraj Jitindar Singh 	assert(prop);
68d129d8a1SSuraj Jitindar Singh 
69d129d8a1SSuraj Jitindar Singh 	nr_threads = len >> 2; /* Divide by 4 since 4 bytes per thread */
70d129d8a1SSuraj Jitindar Singh 	threads = (u32 *)prop->data; /* Array of valid ids */
71d129d8a1SSuraj Jitindar Singh 
72d129d8a1SSuraj Jitindar Singh 	for (i = 0; i < nr_threads; i++) {
73d129d8a1SSuraj Jitindar Singh 		if (!start_thread(fdt32_to_cpu(threads[i]), entry, r3))
74d129d8a1SSuraj Jitindar Singh 			nr_started++;
75d129d8a1SSuraj Jitindar Singh 	}
76d129d8a1SSuraj Jitindar Singh 
77d129d8a1SSuraj Jitindar Singh 	return (struct start_threads) { nr_threads, nr_started };
78d129d8a1SSuraj Jitindar Singh }
79d129d8a1SSuraj Jitindar Singh 
807a20b74eSAndrew Jones static void start_each_secondary(int fdtnode, u64 regval __unused, void *info)
81d129d8a1SSuraj Jitindar Singh {
82d129d8a1SSuraj Jitindar Singh 	struct secondary_entry_data *datap = info;
83d129d8a1SSuraj Jitindar Singh 	struct start_threads ret = start_cpu(fdtnode, datap->entry, datap->r3);
84d129d8a1SSuraj Jitindar Singh 
85d129d8a1SSuraj Jitindar Singh 	nr_threads += ret.nr_threads;
86d129d8a1SSuraj Jitindar Singh 	datap->nr_started += ret.nr_started;
87d129d8a1SSuraj Jitindar Singh }
88d129d8a1SSuraj Jitindar Singh 
89d129d8a1SSuraj Jitindar Singh /*
90d129d8a1SSuraj Jitindar Singh  * Start all stopped cpus on the guest at entry with register 3 set to r3
91d129d8a1SSuraj Jitindar Singh  * We expect that we come in with only one thread currently started
92d129d8a1SSuraj Jitindar Singh  * Returns:	TRUE on success
93d129d8a1SSuraj Jitindar Singh  *		FALSE on failure
94d129d8a1SSuraj Jitindar Singh  */
95d129d8a1SSuraj Jitindar Singh bool start_all_cpus(secondary_entry_fn entry, uint32_t r3)
96d129d8a1SSuraj Jitindar Singh {
97d129d8a1SSuraj Jitindar Singh 	struct secondary_entry_data data = { entry, r3,	0 };
98d129d8a1SSuraj Jitindar Singh 	int ret;
99d129d8a1SSuraj Jitindar Singh 
100d129d8a1SSuraj Jitindar Singh 	ret = dt_for_each_cpu_node(start_each_secondary, &data);
101d129d8a1SSuraj Jitindar Singh 	assert(ret == 0);
102d129d8a1SSuraj Jitindar Singh 
103d129d8a1SSuraj Jitindar Singh 	/* We expect that we come in with one thread already started */
104d129d8a1SSuraj Jitindar Singh 	return data.nr_started == nr_threads - 1;
105d129d8a1SSuraj Jitindar Singh }
106