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