| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  |  * QTest testcase for CPU plugging | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2015 SUSE Linux GmbH | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This work is licensed under the terms of the GNU GPL, version 2 or later. | 
					
						
							|  |  |  |  * See the COPYING file in the top-level directory. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-08 18:08:51 +00:00
										 |  |  | #include "qemu/osdep.h"
 | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include "qemu-common.h"
 | 
					
						
							|  |  |  | #include "libqtest.h"
 | 
					
						
							| 
									
										
										
										
											2018-02-01 12:18:39 +01:00
										 |  |  | #include "qapi/qmp/qdict.h"
 | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  | struct PlugTestData { | 
					
						
							| 
									
										
										
										
											2016-07-18 14:56:51 +04:00
										 |  |  |     char *machine; | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  |     const char *cpu_model; | 
					
						
							| 
									
										
										
										
											2017-10-06 13:32:13 +02:00
										 |  |  |     char *device_model; | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  |     unsigned sockets; | 
					
						
							|  |  |  |     unsigned cores; | 
					
						
							|  |  |  |     unsigned threads; | 
					
						
							|  |  |  |     unsigned maxcpus; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  | typedef struct PlugTestData PlugTestData; | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  | static void test_plug_with_cpu_add(gconstpointer data) | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  |     const PlugTestData *s = data; | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  |     char *args; | 
					
						
							|  |  |  |     QDict *response; | 
					
						
							|  |  |  |     unsigned int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     args = g_strdup_printf("-machine %s -cpu %s " | 
					
						
							| 
									
										
										
										
											2018-09-13 13:06:01 +02:00
										 |  |  |                            "-smp 1,sockets=%u,cores=%u,threads=%u,maxcpus=%u", | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  |                            s->machine, s->cpu_model, | 
					
						
							|  |  |  |                            s->sockets, s->cores, s->threads, s->maxcpus); | 
					
						
							|  |  |  |     qtest_start(args); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 13:06:01 +02:00
										 |  |  |     for (i = 1; i < s->maxcpus; i++) { | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  |         response = qmp("{ 'execute': 'cpu-add'," | 
					
						
							|  |  |  |                        "  'arguments': { 'id': %d } }", i); | 
					
						
							|  |  |  |         g_assert(response); | 
					
						
							|  |  |  |         g_assert(!qdict_haskey(response, "error")); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:01:43 +02:00
										 |  |  |         qobject_unref(response); | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qtest_end(); | 
					
						
							|  |  |  |     g_free(args); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  | static void test_plug_without_cpu_add(gconstpointer data) | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  |     const PlugTestData *s = data; | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  |     char *args; | 
					
						
							|  |  |  |     QDict *response; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     args = g_strdup_printf("-machine %s -cpu %s " | 
					
						
							| 
									
										
										
										
											2018-09-13 13:06:01 +02:00
										 |  |  |                            "-smp 1,sockets=%u,cores=%u,threads=%u,maxcpus=%u", | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  |                            s->machine, s->cpu_model, | 
					
						
							|  |  |  |                            s->sockets, s->cores, s->threads, s->maxcpus); | 
					
						
							|  |  |  |     qtest_start(args); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     response = qmp("{ 'execute': 'cpu-add'," | 
					
						
							|  |  |  |                    "  'arguments': { 'id': %d } }", | 
					
						
							|  |  |  |                    s->sockets * s->cores * s->threads); | 
					
						
							|  |  |  |     g_assert(response); | 
					
						
							|  |  |  |     g_assert(qdict_haskey(response, "error")); | 
					
						
							| 
									
										
										
										
											2018-04-19 17:01:43 +02:00
										 |  |  |     qobject_unref(response); | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     qtest_end(); | 
					
						
							|  |  |  |     g_free(args); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-06 13:32:13 +02:00
										 |  |  | static void test_plug_with_device_add_x86(gconstpointer data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const PlugTestData *td = data; | 
					
						
							|  |  |  |     char *args; | 
					
						
							|  |  |  |     unsigned int s, c, t; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     args = g_strdup_printf("-machine %s -cpu %s " | 
					
						
							| 
									
										
										
										
											2018-09-13 13:06:01 +02:00
										 |  |  |                            "-smp 1,sockets=%u,cores=%u,threads=%u,maxcpus=%u", | 
					
						
							| 
									
										
										
										
											2017-10-06 13:32:13 +02:00
										 |  |  |                            td->machine, td->cpu_model, | 
					
						
							|  |  |  |                            td->sockets, td->cores, td->threads, td->maxcpus); | 
					
						
							|  |  |  |     qtest_start(args); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 13:06:01 +02:00
										 |  |  |     for (s = 1; s < td->sockets; s++) { | 
					
						
							| 
									
										
										
										
											2017-10-06 13:32:13 +02:00
										 |  |  |         for (c = 0; c < td->cores; c++) { | 
					
						
							|  |  |  |             for (t = 0; t < td->threads; t++) { | 
					
						
							|  |  |  |                 char *id = g_strdup_printf("id-%i-%i-%i", s, c, t); | 
					
						
							| 
									
										
										
										
											2018-08-06 08:53:35 +02:00
										 |  |  |                 qtest_qmp_device_add(td->device_model, id, | 
					
						
							|  |  |  |                                      "{'socket-id':%u, 'core-id':%u," | 
					
						
							|  |  |  |                                      " 'thread-id':%u}", | 
					
						
							| 
									
										
										
										
											2017-10-06 13:32:13 +02:00
										 |  |  |                                      s, c, t); | 
					
						
							|  |  |  |                 g_free(id); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qtest_end(); | 
					
						
							|  |  |  |     g_free(args); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-06 15:53:19 +02:00
										 |  |  | static void test_plug_with_device_add_coreid(gconstpointer data) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const PlugTestData *td = data; | 
					
						
							|  |  |  |     char *args; | 
					
						
							|  |  |  |     unsigned int c; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     args = g_strdup_printf("-machine %s -cpu %s " | 
					
						
							|  |  |  |                            "-smp 1,sockets=%u,cores=%u,threads=%u,maxcpus=%u", | 
					
						
							|  |  |  |                            td->machine, td->cpu_model, | 
					
						
							|  |  |  |                            td->sockets, td->cores, td->threads, td->maxcpus); | 
					
						
							|  |  |  |     qtest_start(args); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-13 13:06:01 +02:00
										 |  |  |     for (c = 1; c < td->cores; c++) { | 
					
						
							| 
									
										
										
										
											2017-10-06 15:53:19 +02:00
										 |  |  |         char *id = g_strdup_printf("id-%i", c); | 
					
						
							| 
									
										
										
										
											2018-08-06 08:53:35 +02:00
										 |  |  |         qtest_qmp_device_add(td->device_model, id, "{'core-id':%u}", c); | 
					
						
							| 
									
										
										
										
											2017-10-06 15:53:19 +02:00
										 |  |  |         g_free(id); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     qtest_end(); | 
					
						
							|  |  |  |     g_free(args); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-18 14:56:51 +04:00
										 |  |  | static void test_data_free(gpointer data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  |     PlugTestData *pc = data; | 
					
						
							| 
									
										
										
										
											2016-07-18 14:56:51 +04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     g_free(pc->machine); | 
					
						
							| 
									
										
										
										
											2017-10-06 13:32:13 +02:00
										 |  |  |     g_free(pc->device_model); | 
					
						
							| 
									
										
										
										
											2016-07-18 14:56:51 +04:00
										 |  |  |     g_free(pc); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-30 09:50:06 +02:00
										 |  |  | static void add_pc_test_case(const char *mname) | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2016-07-18 14:56:51 +04:00
										 |  |  |     char *path; | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  |     PlugTestData *data; | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-30 09:50:06 +02:00
										 |  |  |     if (!g_str_has_prefix(mname, "pc-")) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  |     data = g_new(PlugTestData, 1); | 
					
						
							| 
									
										
										
										
											2017-03-30 09:50:06 +02:00
										 |  |  |     data->machine = g_strdup(mname); | 
					
						
							|  |  |  |     data->cpu_model = "Haswell"; /* 1.3+ theoretically */ | 
					
						
							| 
									
										
										
										
											2017-10-06 13:32:13 +02:00
										 |  |  |     data->device_model = g_strdup_printf("%s-%s-cpu", data->cpu_model, | 
					
						
							|  |  |  |                                          qtest_get_arch()); | 
					
						
							| 
									
										
										
										
											2017-03-30 09:50:06 +02:00
										 |  |  |     data->sockets = 1; | 
					
						
							|  |  |  |     data->cores = 3; | 
					
						
							|  |  |  |     data->threads = 2; | 
					
						
							| 
									
										
										
										
											2018-09-13 13:06:01 +02:00
										 |  |  |     data->maxcpus = data->sockets * data->cores * data->threads; | 
					
						
							| 
									
										
										
										
											2017-03-30 09:50:06 +02:00
										 |  |  |     if (g_str_has_suffix(mname, "-1.4") || | 
					
						
							|  |  |  |         (strcmp(mname, "pc-1.3") == 0) || | 
					
						
							|  |  |  |         (strcmp(mname, "pc-1.2") == 0) || | 
					
						
							|  |  |  |         (strcmp(mname, "pc-1.1") == 0) || | 
					
						
							|  |  |  |         (strcmp(mname, "pc-1.0") == 0) || | 
					
						
							|  |  |  |         (strcmp(mname, "pc-0.15") == 0) || | 
					
						
							|  |  |  |         (strcmp(mname, "pc-0.14") == 0) || | 
					
						
							|  |  |  |         (strcmp(mname, "pc-0.13") == 0) || | 
					
						
							|  |  |  |         (strcmp(mname, "pc-0.12") == 0) || | 
					
						
							|  |  |  |         (strcmp(mname, "pc-0.11") == 0) || | 
					
						
							|  |  |  |         (strcmp(mname, "pc-0.10") == 0)) { | 
					
						
							| 
									
										
										
										
											2017-10-06 13:32:13 +02:00
										 |  |  |         path = g_strdup_printf("cpu-plug/%s/init/%ux%ux%u&maxcpus=%u", | 
					
						
							| 
									
										
										
										
											2017-03-30 09:50:06 +02:00
										 |  |  |                                mname, data->sockets, data->cores, | 
					
						
							|  |  |  |                                data->threads, data->maxcpus); | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  |         qtest_add_data_func_full(path, data, test_plug_without_cpu_add, | 
					
						
							| 
									
										
										
										
											2017-03-30 09:50:06 +02:00
										 |  |  |                                  test_data_free); | 
					
						
							|  |  |  |         g_free(path); | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2017-10-06 13:32:13 +02:00
										 |  |  |         PlugTestData *data2 = g_memdup(data, sizeof(PlugTestData)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         data2->machine = g_strdup(data->machine); | 
					
						
							|  |  |  |         data2->device_model = g_strdup(data->device_model); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         path = g_strdup_printf("cpu-plug/%s/cpu-add/%ux%ux%u&maxcpus=%u", | 
					
						
							| 
									
										
										
										
											2017-03-30 09:50:06 +02:00
										 |  |  |                                mname, data->sockets, data->cores, | 
					
						
							|  |  |  |                                data->threads, data->maxcpus); | 
					
						
							| 
									
										
										
										
											2017-10-06 12:13:51 +02:00
										 |  |  |         qtest_add_data_func_full(path, data, test_plug_with_cpu_add, | 
					
						
							| 
									
										
										
										
											2017-03-30 09:50:06 +02:00
										 |  |  |                                  test_data_free); | 
					
						
							|  |  |  |         g_free(path); | 
					
						
							| 
									
										
										
										
											2017-10-06 13:32:13 +02:00
										 |  |  |         path = g_strdup_printf("cpu-plug/%s/device-add/%ux%ux%u&maxcpus=%u", | 
					
						
							|  |  |  |                                mname, data2->sockets, data2->cores, | 
					
						
							|  |  |  |                                data2->threads, data2->maxcpus); | 
					
						
							|  |  |  |         qtest_add_data_func_full(path, data2, test_plug_with_device_add_x86, | 
					
						
							|  |  |  |                                  test_data_free); | 
					
						
							|  |  |  |         g_free(path); | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-06 15:53:19 +02:00
										 |  |  | static void add_pseries_test_case(const char *mname) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     char *path; | 
					
						
							|  |  |  |     PlugTestData *data; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!g_str_has_prefix(mname, "pseries-") || | 
					
						
							|  |  |  |         (g_str_has_prefix(mname, "pseries-2.") && atoi(&mname[10]) < 7)) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     data = g_new(PlugTestData, 1); | 
					
						
							|  |  |  |     data->machine = g_strdup(mname); | 
					
						
							|  |  |  |     data->cpu_model = "power8_v2.0"; | 
					
						
							|  |  |  |     data->device_model = g_strdup("power8_v2.0-spapr-cpu-core"); | 
					
						
							|  |  |  |     data->sockets = 2; | 
					
						
							|  |  |  |     data->cores = 3; | 
					
						
							|  |  |  |     data->threads = 1; | 
					
						
							| 
									
										
										
										
											2018-09-13 13:06:01 +02:00
										 |  |  |     data->maxcpus = data->sockets * data->cores * data->threads; | 
					
						
							| 
									
										
										
										
											2017-10-06 15:53:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     path = g_strdup_printf("cpu-plug/%s/device-add/%ux%ux%u&maxcpus=%u", | 
					
						
							|  |  |  |                            mname, data->sockets, data->cores, | 
					
						
							|  |  |  |                            data->threads, data->maxcpus); | 
					
						
							|  |  |  |     qtest_add_data_func_full(path, data, test_plug_with_device_add_coreid, | 
					
						
							|  |  |  |                              test_data_free); | 
					
						
							|  |  |  |     g_free(path); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-17 13:39:15 +02:00
										 |  |  | static void add_s390x_test_case(const char *mname) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     char *path; | 
					
						
							|  |  |  |     PlugTestData *data, *data2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!g_str_has_prefix(mname, "s390-ccw-virtio-")) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     data = g_new(PlugTestData, 1); | 
					
						
							|  |  |  |     data->machine = g_strdup(mname); | 
					
						
							|  |  |  |     data->cpu_model = "qemu"; | 
					
						
							|  |  |  |     data->device_model = g_strdup("qemu-s390x-cpu"); | 
					
						
							|  |  |  |     data->sockets = 1; | 
					
						
							|  |  |  |     data->cores = 3; | 
					
						
							|  |  |  |     data->threads = 1; | 
					
						
							| 
									
										
										
										
											2018-09-13 13:06:01 +02:00
										 |  |  |     data->maxcpus = data->sockets * data->cores * data->threads; | 
					
						
							| 
									
										
										
										
											2017-10-17 13:39:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     data2 = g_memdup(data, sizeof(PlugTestData)); | 
					
						
							|  |  |  |     data2->machine = g_strdup(data->machine); | 
					
						
							|  |  |  |     data2->device_model = g_strdup(data->device_model); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     path = g_strdup_printf("cpu-plug/%s/cpu-add/%ux%ux%u&maxcpus=%u", | 
					
						
							|  |  |  |                            mname, data->sockets, data->cores, | 
					
						
							|  |  |  |                            data->threads, data->maxcpus); | 
					
						
							|  |  |  |     qtest_add_data_func_full(path, data, test_plug_with_cpu_add, | 
					
						
							|  |  |  |                              test_data_free); | 
					
						
							|  |  |  |     g_free(path); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     path = g_strdup_printf("cpu-plug/%s/device-add/%ux%ux%u&maxcpus=%u", | 
					
						
							|  |  |  |                            mname, data2->sockets, data2->cores, | 
					
						
							|  |  |  |                            data2->threads, data2->maxcpus); | 
					
						
							|  |  |  |     qtest_add_data_func_full(path, data2, test_plug_with_device_add_coreid, | 
					
						
							|  |  |  |                              test_data_free); | 
					
						
							|  |  |  |     g_free(path); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  | int main(int argc, char **argv) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const char *arch = qtest_get_arch(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     g_test_init(&argc, &argv, NULL); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (strcmp(arch, "i386") == 0 || strcmp(arch, "x86_64") == 0) { | 
					
						
							| 
									
										
										
										
											2018-08-16 13:35:55 +02:00
										 |  |  |         qtest_cb_for_every_machine(add_pc_test_case, g_test_quick()); | 
					
						
							| 
									
										
										
										
											2017-10-06 15:53:19 +02:00
										 |  |  |     } else if (g_str_equal(arch, "ppc64")) { | 
					
						
							| 
									
										
										
										
											2018-08-16 13:35:55 +02:00
										 |  |  |         qtest_cb_for_every_machine(add_pseries_test_case, g_test_quick()); | 
					
						
							| 
									
										
										
										
											2017-10-17 13:39:15 +02:00
										 |  |  |     } else if (g_str_equal(arch, "s390x")) { | 
					
						
							| 
									
										
										
										
											2018-08-16 13:35:55 +02:00
										 |  |  |         qtest_cb_for_every_machine(add_s390x_test_case, g_test_quick()); | 
					
						
							| 
									
										
										
										
											2015-03-13 17:21:11 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return g_test_run(); | 
					
						
							|  |  |  | } |