289f54ac307a941ef1456944fc8a98491f90ae88
[AGL/meta-agl-devel.git] /
1 From faa349f4d096554c6d5bfe74634599d2e26f64a7 Mon Sep 17 00:00:00 2001
2 From: Jan Kiszka <jan.kiszka@siemens.com>
3 Date: Sun, 11 Sep 2016 23:30:04 +0200
4 Subject: [PATCH 03/32] jailhouse: Add simple debug console via the hypervisor
5
6 Jailhouse allows explicitly enabled cells to write character-wise
7 messages to the hypervisor debug console. Make use of this for a
8 platform-agnostic boot diagnosis channel, specifically for non-root
9 cells. This also comes with earlycon support.
10
11 Signed-off-by: Jan Kiszka <jan.kiszka@siemens.com>
12 ---
13  MAINTAINERS                     |   1 +
14  drivers/virt/Kconfig            |  11 +++++
15  drivers/virt/Makefile           |   1 +
16  drivers/virt/jailhouse_dbgcon.c | 103 ++++++++++++++++++++++++++++++++++++++++
17  4 files changed, 116 insertions(+)
18  create mode 100644 drivers/virt/jailhouse_dbgcon.c
19
20 diff --git a/MAINTAINERS b/MAINTAINERS
21 index 9d3a5c54a41d..07cb4d674c93 100644
22 --- a/MAINTAINERS
23 +++ b/MAINTAINERS
24 @@ -8761,6 +8761,7 @@ L:        jailhouse-dev@googlegroups.com
25  S:     Maintained
26  F:     arch/x86/kernel/jailhouse.c
27  F:     arch/x86/include/asm/jailhouse_para.h
28 +F:     drivers/virt/jailhouse_dbgcon.c
29  
30  JC42.4 TEMPERATURE SENSOR DRIVER
31  M:     Guenter Roeck <linux@roeck-us.net>
32 diff --git a/drivers/virt/Kconfig b/drivers/virt/Kconfig
33 index 363af2eaf2ba..99c5eaca6952 100644
34 --- a/drivers/virt/Kconfig
35 +++ b/drivers/virt/Kconfig
36 @@ -31,5 +31,16 @@ config FSL_HV_MANAGER
37            4) A kernel interface for receiving callbacks when a managed
38              partition shuts down.
39  
40 +config JAILHOUSE_DBGCON
41 +       tristate "Jailhouse console driver"
42 +       depends on X86 || ARM || ARM64
43 +       help
44 +         The Jailhouse hypervisor provides a simple write-only console for
45 +         debugging the bootstrap process of its cells. This driver registers
46 +         a console with the kernel to make use of it.
47 +
48 +         Note that Jailhouse has to be configured to permit a cell the usage
49 +         of the console interface.
50 +
51  source "drivers/virt/vboxguest/Kconfig"
52  endif
53 diff --git a/drivers/virt/Makefile b/drivers/virt/Makefile
54 index fd331247c27a..89e86a1d0f19 100644
55 --- a/drivers/virt/Makefile
56 +++ b/drivers/virt/Makefile
57 @@ -4,4 +4,5 @@
58  #
59  
60  obj-$(CONFIG_FSL_HV_MANAGER)   += fsl_hypervisor.o
61 +obj-$(CONFIG_JAILHOUSE_DBGCON) += jailhouse_dbgcon.o
62  obj-y                          += vboxguest/
63 diff --git a/drivers/virt/jailhouse_dbgcon.c b/drivers/virt/jailhouse_dbgcon.c
64 new file mode 100644
65 index 000000000000..1fd201ea1460
66 --- /dev/null
67 +++ b/drivers/virt/jailhouse_dbgcon.c
68 @@ -0,0 +1,103 @@
69 +/* SPDX-License-Identifier: GPL-2.0 */
70 +/*
71 + * Console driver for running over the Jailhouse partitioning hypervisor
72 + *
73 + * Copyright (c) Siemens AG, 2016-2018
74 + *
75 + * Authors:
76 + *  Jan Kiszka <jan.kiszka@siemens.com>
77 + */
78 +
79 +#include <linux/console.h>
80 +#include <linux/hypervisor.h>
81 +#include <linux/module.h>
82 +#include <linux/serial_core.h>
83 +#ifdef CONFIG_X86
84 +#include <asm/alternative.h>
85 +#endif
86 +#ifdef CONFIG_ARM
87 +#include <asm/opcodes-virt.h>
88 +#endif
89 +
90 +#define JAILHOUSE_HC_DEBUG_CONSOLE_PUTC                8
91 +
92 +static void hypervisor_putc(char c)
93 +{
94 +#if defined(CONFIG_X86)
95 +       int result;
96 +
97 +       asm volatile(
98 +               ALTERNATIVE(".byte 0x0f,0x01,0xc1", ".byte 0x0f,0x01,0xd9",
99 +                           X86_FEATURE_VMMCALL)
100 +               : "=a" (result)
101 +               : "a" (JAILHOUSE_HC_DEBUG_CONSOLE_PUTC), "D" (c)
102 +               : "memory");
103 +#elif defined(CONFIG_ARM)
104 +       register u32 num_res asm("r0") = JAILHOUSE_HC_DEBUG_CONSOLE_PUTC;
105 +       register u32 arg1 asm("r1") = c;
106 +
107 +       asm volatile(
108 +               __HVC(0x4a48)
109 +               : "=r" (num_res)
110 +               : "r" (num_res), "r" (arg1)
111 +               : "memory");
112 +#elif defined(CONFIG_ARM64)
113 +       register u64 num_res asm("x0") = JAILHOUSE_HC_DEBUG_CONSOLE_PUTC;
114 +       register u64 arg1 asm("x1") = c;
115 +
116 +       asm volatile(
117 +               "hvc #0x4a48\n\t"
118 +               : "=r" (num_res)
119 +               : "r" (num_res), "r" (arg1)
120 +               : "memory");
121 +#else
122 +#error Unsupported architecture.
123 +#endif
124 +}
125 +
126 +static void jailhouse_dbgcon_write(struct console *con, const char *s,
127 +                                  unsigned count)
128 +{
129 +       while (count > 0) {
130 +               hypervisor_putc(*s);
131 +               count--;
132 +               s++;
133 +       }
134 +}
135 +
136 +static int __init early_jailhouse_dbgcon_setup(struct earlycon_device *device,
137 +                                              const char *options)
138 +{
139 +       device->con->write = jailhouse_dbgcon_write;
140 +       return 0;
141 +}
142 +
143 +EARLYCON_DECLARE(jailhouse, early_jailhouse_dbgcon_setup);
144 +
145 +static struct console jailhouse_dbgcon = {
146 +       .name = "jailhouse",
147 +       .write = jailhouse_dbgcon_write,
148 +       .flags = CON_PRINTBUFFER | CON_ANYTIME,
149 +       .index = -1,
150 +};
151 +
152 +static int __init jailhouse_dbgcon_init(void)
153 +{
154 +       if (!jailhouse_paravirt())
155 +               return -ENODEV;
156 +
157 +       register_console(&jailhouse_dbgcon);
158 +       return 0;
159 +}
160 +
161 +static void __exit jailhouse_dbgcon_exit(void)
162 +{
163 +       unregister_console(&jailhouse_dbgcon);
164 +}
165 +
166 +module_init(jailhouse_dbgcon_init);
167 +module_exit(jailhouse_dbgcon_exit);
168 +
169 +MODULE_LICENSE("GPL v2");
170 +MODULE_DESCRIPTION("Jailhouse debug console driver");
171 +MODULE_AUTHOR("Jan Kiszka <jan.kiszka@siemens.com>");
172 -- 
173 2.11.0
174