1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26#include "sysemu/accel.h"
27#include "hw/boards.h"
28#include "qemu-common.h"
29#include "sysemu/arch_init.h"
30#include "sysemu/sysemu.h"
31#include "sysemu/kvm.h"
32#include "sysemu/qtest.h"
33#include "hw/xen/xen.h"
34#include "qom/object.h"
35#include "hw/boards.h"
36
37int tcg_tb_size;
38static bool tcg_allowed = true;
39
40static int tcg_init(MachineState *ms)
41{
42 tcg_exec_init(tcg_tb_size * 1024 * 1024);
43 return 0;
44}
45
46static const TypeInfo accel_type = {
47 .name = TYPE_ACCEL,
48 .parent = TYPE_OBJECT,
49 .class_size = sizeof(AccelClass),
50 .instance_size = sizeof(AccelState),
51};
52
53
54static AccelClass *accel_find(const char *opt_name)
55{
56 char *class_name = g_strdup_printf(ACCEL_CLASS_NAME("%s"), opt_name);
57 AccelClass *ac = ACCEL_CLASS(object_class_by_name(class_name));
58 g_free(class_name);
59 return ac;
60}
61
62static int accel_init_machine(AccelClass *acc, MachineState *ms)
63{
64 ObjectClass *oc = OBJECT_CLASS(acc);
65 const char *cname = object_class_get_name(oc);
66 AccelState *accel = ACCEL(object_new(cname));
67 int ret;
68 ms->accelerator = accel;
69 *(acc->allowed) = true;
70 ret = acc->init_machine(ms);
71 if (ret < 0) {
72 ms->accelerator = NULL;
73 *(acc->allowed) = false;
74 object_unref(OBJECT(accel));
75 }
76 return ret;
77}
78
79int configure_accelerator(MachineState *ms)
80{
81 const char *p;
82 char buf[10];
83 int ret;
84 bool accel_initialised = false;
85 bool init_failed = false;
86 AccelClass *acc = NULL;
87
88 p = qemu_opt_get(qemu_get_machine_opts(), "accel");
89 if (p == NULL) {
90
91 p = "tcg";
92 }
93
94 while (!accel_initialised && *p != '\0') {
95 if (*p == ':') {
96 p++;
97 }
98 p = get_opt_name(buf, sizeof(buf), p, ':');
99 acc = accel_find(buf);
100 if (!acc) {
101 fprintf(stderr, "\"%s\" accelerator not found.\n", buf);
102 continue;
103 }
104 if (acc->available && !acc->available()) {
105 printf("%s not supported for this target\n",
106 acc->name);
107 continue;
108 }
109 ret = accel_init_machine(acc, ms);
110 if (ret < 0) {
111 init_failed = true;
112 fprintf(stderr, "failed to initialize %s: %s\n",
113 acc->name,
114 strerror(-ret));
115 } else {
116 accel_initialised = true;
117 }
118 }
119
120 if (!accel_initialised) {
121 if (!init_failed) {
122 fprintf(stderr, "No accelerator found!\n");
123 }
124 exit(1);
125 }
126
127 if (init_failed) {
128 fprintf(stderr, "Back to %s accelerator.\n", acc->name);
129 }
130
131 return !accel_initialised;
132}
133
134
135static void tcg_accel_class_init(ObjectClass *oc, void *data)
136{
137 AccelClass *ac = ACCEL_CLASS(oc);
138 ac->name = "tcg";
139 ac->init_machine = tcg_init;
140 ac->allowed = &tcg_allowed;
141}
142
143#define TYPE_TCG_ACCEL ACCEL_CLASS_NAME("tcg")
144
145static const TypeInfo tcg_accel_type = {
146 .name = TYPE_TCG_ACCEL,
147 .parent = TYPE_ACCEL,
148 .class_init = tcg_accel_class_init,
149};
150
151static void register_accel_types(void)
152{
153 type_register_static(&accel_type);
154 type_register_static(&tcg_accel_type);
155}
156
157type_init(register_accel_types);
158