diff options
author | Jeremy Fitzhardinge <jeremy.fitzhardinge@citrix.com> | 2010-09-02 11:17:06 -0400 |
---|---|---|
committer | Stefano Stabellini <stefano.stabellini@eu.citrix.com> | 2010-10-22 16:26:01 -0400 |
commit | 4fe7d5a708a955b35e3fdc4dea3e0b7a6ae2eb06 (patch) | |
tree | 43fff2734caa2efa8d7adc8c3439d4c282f29081 | |
parent | 4ec5387cc36c6472a2ff2c82e9865abe8cab96c2 (diff) |
xen: make hvc_xen console work for dom0.
Use the console hypercalls for dom0 console.
[ Impact: Add Xen dom0 console ]
Signed-off-by: Jeremy Fitzhardinge <jeremy.fitzhardinge@citrix.com>
Signed-off-by: Juan Quintela <quintela@redhat.com>
Signed-off-by: Stefano Stabellini <stefano.stabellini@eu.citrix.com>
Reviewed-by: Konrad Rzeszutek Wilk <konrad.wilk@oracle.com>
-rw-r--r-- | drivers/char/hvc_xen.c | 98 | ||||
-rw-r--r-- | drivers/xen/events.c | 2 | ||||
-rw-r--r-- | include/xen/events.h | 1 |
3 files changed, 67 insertions, 34 deletions
diff --git a/drivers/char/hvc_xen.c b/drivers/char/hvc_xen.c index 60446f82a3fc..1f7e13a43a88 100644 --- a/drivers/char/hvc_xen.c +++ b/drivers/char/hvc_xen.c | |||
@@ -78,7 +78,7 @@ static int __write_console(const char *data, int len) | |||
78 | return sent; | 78 | return sent; |
79 | } | 79 | } |
80 | 80 | ||
81 | static int write_console(uint32_t vtermno, const char *data, int len) | 81 | static int domU_write_console(uint32_t vtermno, const char *data, int len) |
82 | { | 82 | { |
83 | int ret = len; | 83 | int ret = len; |
84 | 84 | ||
@@ -101,7 +101,7 @@ static int write_console(uint32_t vtermno, const char *data, int len) | |||
101 | return ret; | 101 | return ret; |
102 | } | 102 | } |
103 | 103 | ||
104 | static int read_console(uint32_t vtermno, char *buf, int len) | 104 | static int domU_read_console(uint32_t vtermno, char *buf, int len) |
105 | { | 105 | { |
106 | struct xencons_interface *intf = xencons_interface(); | 106 | struct xencons_interface *intf = xencons_interface(); |
107 | XENCONS_RING_IDX cons, prod; | 107 | XENCONS_RING_IDX cons, prod; |
@@ -122,28 +122,62 @@ static int read_console(uint32_t vtermno, char *buf, int len) | |||
122 | return recv; | 122 | return recv; |
123 | } | 123 | } |
124 | 124 | ||
125 | static const struct hv_ops hvc_ops = { | 125 | static struct hv_ops domU_hvc_ops = { |
126 | .get_chars = read_console, | 126 | .get_chars = domU_read_console, |
127 | .put_chars = write_console, | 127 | .put_chars = domU_write_console, |
128 | .notifier_add = notifier_add_irq, | 128 | .notifier_add = notifier_add_irq, |
129 | .notifier_del = notifier_del_irq, | 129 | .notifier_del = notifier_del_irq, |
130 | .notifier_hangup = notifier_hangup_irq, | 130 | .notifier_hangup = notifier_hangup_irq, |
131 | }; | 131 | }; |
132 | 132 | ||
133 | static int __init xen_init(void) | 133 | static int dom0_read_console(uint32_t vtermno, char *buf, int len) |
134 | { | ||
135 | return HYPERVISOR_console_io(CONSOLEIO_read, len, buf); | ||
136 | } | ||
137 | |||
138 | /* | ||
139 | * Either for a dom0 to write to the system console, or a domU with a | ||
140 | * debug version of Xen | ||
141 | */ | ||
142 | static int dom0_write_console(uint32_t vtermno, const char *str, int len) | ||
143 | { | ||
144 | int rc = HYPERVISOR_console_io(CONSOLEIO_write, len, (char *)str); | ||
145 | if (rc < 0) | ||
146 | return 0; | ||
147 | |||
148 | return len; | ||
149 | } | ||
150 | |||
151 | static struct hv_ops dom0_hvc_ops = { | ||
152 | .get_chars = dom0_read_console, | ||
153 | .put_chars = dom0_write_console, | ||
154 | .notifier_add = notifier_add_irq, | ||
155 | .notifier_del = notifier_del_irq, | ||
156 | .notifier_hangup = notifier_hangup_irq, | ||
157 | }; | ||
158 | |||
159 | static int __init xen_hvc_init(void) | ||
134 | { | 160 | { |
135 | struct hvc_struct *hp; | 161 | struct hvc_struct *hp; |
162 | struct hv_ops *ops; | ||
136 | 163 | ||
137 | if (!xen_pv_domain() || | 164 | if (!xen_pv_domain()) |
138 | xen_initial_domain() || | ||
139 | !xen_start_info->console.domU.evtchn) | ||
140 | return -ENODEV; | 165 | return -ENODEV; |
141 | 166 | ||
142 | xencons_irq = bind_evtchn_to_irq(xen_start_info->console.domU.evtchn); | 167 | if (xen_initial_domain()) { |
168 | ops = &dom0_hvc_ops; | ||
169 | xencons_irq = bind_virq_to_irq(VIRQ_CONSOLE, 0); | ||
170 | } else { | ||
171 | if (!xen_start_info->console.domU.evtchn) | ||
172 | return -ENODEV; | ||
173 | |||
174 | ops = &domU_hvc_ops; | ||
175 | xencons_irq = bind_evtchn_to_irq(xen_start_info->console.domU.evtchn); | ||
176 | } | ||
143 | if (xencons_irq < 0) | 177 | if (xencons_irq < 0) |
144 | xencons_irq = 0; /* NO_IRQ */ | 178 | xencons_irq = 0; /* NO_IRQ */ |
145 | 179 | ||
146 | hp = hvc_alloc(HVC_COOKIE, xencons_irq, &hvc_ops, 256); | 180 | hp = hvc_alloc(HVC_COOKIE, xencons_irq, ops, 256); |
147 | if (IS_ERR(hp)) | 181 | if (IS_ERR(hp)) |
148 | return PTR_ERR(hp); | 182 | return PTR_ERR(hp); |
149 | 183 | ||
@@ -160,7 +194,7 @@ void xen_console_resume(void) | |||
160 | rebind_evtchn_irq(xen_start_info->console.domU.evtchn, xencons_irq); | 194 | rebind_evtchn_irq(xen_start_info->console.domU.evtchn, xencons_irq); |
161 | } | 195 | } |
162 | 196 | ||
163 | static void __exit xen_fini(void) | 197 | static void __exit xen_hvc_fini(void) |
164 | { | 198 | { |
165 | if (hvc) | 199 | if (hvc) |
166 | hvc_remove(hvc); | 200 | hvc_remove(hvc); |
@@ -168,29 +202,24 @@ static void __exit xen_fini(void) | |||
168 | 202 | ||
169 | static int xen_cons_init(void) | 203 | static int xen_cons_init(void) |
170 | { | 204 | { |
205 | struct hv_ops *ops; | ||
206 | |||
171 | if (!xen_pv_domain()) | 207 | if (!xen_pv_domain()) |
172 | return 0; | 208 | return 0; |
173 | 209 | ||
174 | hvc_instantiate(HVC_COOKIE, 0, &hvc_ops); | 210 | if (xen_initial_domain()) |
211 | ops = &dom0_hvc_ops; | ||
212 | else | ||
213 | ops = &domU_hvc_ops; | ||
214 | |||
215 | hvc_instantiate(HVC_COOKIE, 0, ops); | ||
175 | return 0; | 216 | return 0; |
176 | } | 217 | } |
177 | 218 | ||
178 | module_init(xen_init); | 219 | module_init(xen_hvc_init); |
179 | module_exit(xen_fini); | 220 | module_exit(xen_hvc_fini); |
180 | console_initcall(xen_cons_init); | 221 | console_initcall(xen_cons_init); |
181 | 222 | ||
182 | static void raw_console_write(const char *str, int len) | ||
183 | { | ||
184 | while(len > 0) { | ||
185 | int rc = HYPERVISOR_console_io(CONSOLEIO_write, len, (char *)str); | ||
186 | if (rc <= 0) | ||
187 | break; | ||
188 | |||
189 | str += rc; | ||
190 | len -= rc; | ||
191 | } | ||
192 | } | ||
193 | |||
194 | #ifdef CONFIG_EARLY_PRINTK | 223 | #ifdef CONFIG_EARLY_PRINTK |
195 | static void xenboot_write_console(struct console *console, const char *string, | 224 | static void xenboot_write_console(struct console *console, const char *string, |
196 | unsigned len) | 225 | unsigned len) |
@@ -198,19 +227,22 @@ static void xenboot_write_console(struct console *console, const char *string, | |||
198 | unsigned int linelen, off = 0; | 227 | unsigned int linelen, off = 0; |
199 | const char *pos; | 228 | const char *pos; |
200 | 229 | ||
201 | raw_console_write(string, len); | 230 | dom0_write_console(0, string, len); |
231 | |||
232 | if (xen_initial_domain()) | ||
233 | return; | ||
202 | 234 | ||
203 | write_console(0, "(early) ", 8); | 235 | domU_write_console(0, "(early) ", 8); |
204 | while (off < len && NULL != (pos = strchr(string+off, '\n'))) { | 236 | while (off < len && NULL != (pos = strchr(string+off, '\n'))) { |
205 | linelen = pos-string+off; | 237 | linelen = pos-string+off; |
206 | if (off + linelen > len) | 238 | if (off + linelen > len) |
207 | break; | 239 | break; |
208 | write_console(0, string+off, linelen); | 240 | domU_write_console(0, string+off, linelen); |
209 | write_console(0, "\r\n", 2); | 241 | domU_write_console(0, "\r\n", 2); |
210 | off += linelen + 1; | 242 | off += linelen + 1; |
211 | } | 243 | } |
212 | if (off < len) | 244 | if (off < len) |
213 | write_console(0, string+off, len-off); | 245 | domU_write_console(0, string+off, len-off); |
214 | } | 246 | } |
215 | 247 | ||
216 | struct console xenboot_console = { | 248 | struct console xenboot_console = { |
@@ -222,7 +254,7 @@ struct console xenboot_console = { | |||
222 | 254 | ||
223 | void xen_raw_console_write(const char *str) | 255 | void xen_raw_console_write(const char *str) |
224 | { | 256 | { |
225 | raw_console_write(str, strlen(str)); | 257 | dom0_write_console(0, str, strlen(str)); |
226 | } | 258 | } |
227 | 259 | ||
228 | void xen_raw_printk(const char *fmt, ...) | 260 | void xen_raw_printk(const char *fmt, ...) |
diff --git a/drivers/xen/events.c b/drivers/xen/events.c index a7d9555e664d..93e98ffe71ae 100644 --- a/drivers/xen/events.c +++ b/drivers/xen/events.c | |||
@@ -839,7 +839,7 @@ static int bind_ipi_to_irq(unsigned int ipi, unsigned int cpu) | |||
839 | } | 839 | } |
840 | 840 | ||
841 | 841 | ||
842 | static int bind_virq_to_irq(unsigned int virq, unsigned int cpu) | 842 | int bind_virq_to_irq(unsigned int virq, unsigned int cpu) |
843 | { | 843 | { |
844 | struct evtchn_bind_virq bind_virq; | 844 | struct evtchn_bind_virq bind_virq; |
845 | int evtchn, irq; | 845 | int evtchn, irq; |
diff --git a/include/xen/events.h b/include/xen/events.h index 8fa27dc7358b..646dd17d3aa4 100644 --- a/include/xen/events.h +++ b/include/xen/events.h | |||
@@ -12,6 +12,7 @@ int bind_evtchn_to_irqhandler(unsigned int evtchn, | |||
12 | irq_handler_t handler, | 12 | irq_handler_t handler, |
13 | unsigned long irqflags, const char *devname, | 13 | unsigned long irqflags, const char *devname, |
14 | void *dev_id); | 14 | void *dev_id); |
15 | int bind_virq_to_irq(unsigned int virq, unsigned int cpu); | ||
15 | int bind_virq_to_irqhandler(unsigned int virq, unsigned int cpu, | 16 | int bind_virq_to_irqhandler(unsigned int virq, unsigned int cpu, |
16 | irq_handler_t handler, | 17 | irq_handler_t handler, |
17 | unsigned long irqflags, const char *devname, | 18 | unsigned long irqflags, const char *devname, |