xref: /OK3568_Linux_fs/kernel/drivers/xen/efi.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun  * EFI support for Xen.
4*4882a593Smuzhiyun  *
5*4882a593Smuzhiyun  * Copyright (C) 1999 VA Linux Systems
6*4882a593Smuzhiyun  * Copyright (C) 1999 Walt Drummond <drummond@valinux.com>
7*4882a593Smuzhiyun  * Copyright (C) 1999-2002 Hewlett-Packard Co.
8*4882a593Smuzhiyun  *	David Mosberger-Tang <davidm@hpl.hp.com>
9*4882a593Smuzhiyun  *	Stephane Eranian <eranian@hpl.hp.com>
10*4882a593Smuzhiyun  * Copyright (C) 2005-2008 Intel Co.
11*4882a593Smuzhiyun  *	Fenghua Yu <fenghua.yu@intel.com>
12*4882a593Smuzhiyun  *	Bibo Mao <bibo.mao@intel.com>
13*4882a593Smuzhiyun  *	Chandramouli Narayanan <mouli@linux.intel.com>
14*4882a593Smuzhiyun  *	Huang Ying <ying.huang@intel.com>
15*4882a593Smuzhiyun  * Copyright (C) 2011 Novell Co.
16*4882a593Smuzhiyun  *	Jan Beulich <JBeulich@suse.com>
17*4882a593Smuzhiyun  * Copyright (C) 2011-2012 Oracle Co.
18*4882a593Smuzhiyun  *	Liang Tang <liang.tang@oracle.com>
19*4882a593Smuzhiyun  * Copyright (c) 2014 Oracle Co., Daniel Kiper
20*4882a593Smuzhiyun  */
21*4882a593Smuzhiyun 
22*4882a593Smuzhiyun #include <linux/bug.h>
23*4882a593Smuzhiyun #include <linux/efi.h>
24*4882a593Smuzhiyun #include <linux/init.h>
25*4882a593Smuzhiyun #include <linux/string.h>
26*4882a593Smuzhiyun 
27*4882a593Smuzhiyun #include <xen/interface/xen.h>
28*4882a593Smuzhiyun #include <xen/interface/platform.h>
29*4882a593Smuzhiyun #include <xen/xen.h>
30*4882a593Smuzhiyun #include <xen/xen-ops.h>
31*4882a593Smuzhiyun 
32*4882a593Smuzhiyun #include <asm/page.h>
33*4882a593Smuzhiyun 
34*4882a593Smuzhiyun #include <asm/xen/hypercall.h>
35*4882a593Smuzhiyun 
36*4882a593Smuzhiyun #define INIT_EFI_OP(name) \
37*4882a593Smuzhiyun 	{.cmd = XENPF_efi_runtime_call, \
38*4882a593Smuzhiyun 	 .u.efi_runtime_call.function = XEN_EFI_##name, \
39*4882a593Smuzhiyun 	 .u.efi_runtime_call.misc = 0}
40*4882a593Smuzhiyun 
41*4882a593Smuzhiyun #define efi_data(op)	(op.u.efi_runtime_call)
42*4882a593Smuzhiyun 
xen_efi_get_time(efi_time_t * tm,efi_time_cap_t * tc)43*4882a593Smuzhiyun static efi_status_t xen_efi_get_time(efi_time_t *tm, efi_time_cap_t *tc)
44*4882a593Smuzhiyun {
45*4882a593Smuzhiyun 	struct xen_platform_op op = INIT_EFI_OP(get_time);
46*4882a593Smuzhiyun 
47*4882a593Smuzhiyun 	if (HYPERVISOR_platform_op(&op) < 0)
48*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
49*4882a593Smuzhiyun 
50*4882a593Smuzhiyun 	if (tm) {
51*4882a593Smuzhiyun 		BUILD_BUG_ON(sizeof(*tm) != sizeof(efi_data(op).u.get_time.time));
52*4882a593Smuzhiyun 		memcpy(tm, &efi_data(op).u.get_time.time, sizeof(*tm));
53*4882a593Smuzhiyun 	}
54*4882a593Smuzhiyun 
55*4882a593Smuzhiyun 	if (tc) {
56*4882a593Smuzhiyun 		tc->resolution = efi_data(op).u.get_time.resolution;
57*4882a593Smuzhiyun 		tc->accuracy = efi_data(op).u.get_time.accuracy;
58*4882a593Smuzhiyun 		tc->sets_to_zero = !!(efi_data(op).misc &
59*4882a593Smuzhiyun 				      XEN_EFI_GET_TIME_SET_CLEARS_NS);
60*4882a593Smuzhiyun 	}
61*4882a593Smuzhiyun 
62*4882a593Smuzhiyun 	return efi_data(op).status;
63*4882a593Smuzhiyun }
64*4882a593Smuzhiyun 
xen_efi_set_time(efi_time_t * tm)65*4882a593Smuzhiyun static efi_status_t xen_efi_set_time(efi_time_t *tm)
66*4882a593Smuzhiyun {
67*4882a593Smuzhiyun 	struct xen_platform_op op = INIT_EFI_OP(set_time);
68*4882a593Smuzhiyun 
69*4882a593Smuzhiyun 	BUILD_BUG_ON(sizeof(*tm) != sizeof(efi_data(op).u.set_time));
70*4882a593Smuzhiyun 	memcpy(&efi_data(op).u.set_time, tm, sizeof(*tm));
71*4882a593Smuzhiyun 
72*4882a593Smuzhiyun 	if (HYPERVISOR_platform_op(&op) < 0)
73*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
74*4882a593Smuzhiyun 
75*4882a593Smuzhiyun 	return efi_data(op).status;
76*4882a593Smuzhiyun }
77*4882a593Smuzhiyun 
xen_efi_get_wakeup_time(efi_bool_t * enabled,efi_bool_t * pending,efi_time_t * tm)78*4882a593Smuzhiyun static efi_status_t xen_efi_get_wakeup_time(efi_bool_t *enabled,
79*4882a593Smuzhiyun 					    efi_bool_t *pending,
80*4882a593Smuzhiyun 					    efi_time_t *tm)
81*4882a593Smuzhiyun {
82*4882a593Smuzhiyun 	struct xen_platform_op op = INIT_EFI_OP(get_wakeup_time);
83*4882a593Smuzhiyun 
84*4882a593Smuzhiyun 	if (HYPERVISOR_platform_op(&op) < 0)
85*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
86*4882a593Smuzhiyun 
87*4882a593Smuzhiyun 	if (tm) {
88*4882a593Smuzhiyun 		BUILD_BUG_ON(sizeof(*tm) != sizeof(efi_data(op).u.get_wakeup_time));
89*4882a593Smuzhiyun 		memcpy(tm, &efi_data(op).u.get_wakeup_time, sizeof(*tm));
90*4882a593Smuzhiyun 	}
91*4882a593Smuzhiyun 
92*4882a593Smuzhiyun 	if (enabled)
93*4882a593Smuzhiyun 		*enabled = !!(efi_data(op).misc & XEN_EFI_GET_WAKEUP_TIME_ENABLED);
94*4882a593Smuzhiyun 
95*4882a593Smuzhiyun 	if (pending)
96*4882a593Smuzhiyun 		*pending = !!(efi_data(op).misc & XEN_EFI_GET_WAKEUP_TIME_PENDING);
97*4882a593Smuzhiyun 
98*4882a593Smuzhiyun 	return efi_data(op).status;
99*4882a593Smuzhiyun }
100*4882a593Smuzhiyun 
xen_efi_set_wakeup_time(efi_bool_t enabled,efi_time_t * tm)101*4882a593Smuzhiyun static efi_status_t xen_efi_set_wakeup_time(efi_bool_t enabled, efi_time_t *tm)
102*4882a593Smuzhiyun {
103*4882a593Smuzhiyun 	struct xen_platform_op op = INIT_EFI_OP(set_wakeup_time);
104*4882a593Smuzhiyun 
105*4882a593Smuzhiyun 	BUILD_BUG_ON(sizeof(*tm) != sizeof(efi_data(op).u.set_wakeup_time));
106*4882a593Smuzhiyun 	if (enabled)
107*4882a593Smuzhiyun 		efi_data(op).misc = XEN_EFI_SET_WAKEUP_TIME_ENABLE;
108*4882a593Smuzhiyun 	if (tm)
109*4882a593Smuzhiyun 		memcpy(&efi_data(op).u.set_wakeup_time, tm, sizeof(*tm));
110*4882a593Smuzhiyun 	else
111*4882a593Smuzhiyun 		efi_data(op).misc |= XEN_EFI_SET_WAKEUP_TIME_ENABLE_ONLY;
112*4882a593Smuzhiyun 
113*4882a593Smuzhiyun 	if (HYPERVISOR_platform_op(&op) < 0)
114*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
115*4882a593Smuzhiyun 
116*4882a593Smuzhiyun 	return efi_data(op).status;
117*4882a593Smuzhiyun }
118*4882a593Smuzhiyun 
xen_efi_get_variable(efi_char16_t * name,efi_guid_t * vendor,u32 * attr,unsigned long * data_size,void * data)119*4882a593Smuzhiyun static efi_status_t xen_efi_get_variable(efi_char16_t *name, efi_guid_t *vendor,
120*4882a593Smuzhiyun 					 u32 *attr, unsigned long *data_size,
121*4882a593Smuzhiyun 					 void *data)
122*4882a593Smuzhiyun {
123*4882a593Smuzhiyun 	struct xen_platform_op op = INIT_EFI_OP(get_variable);
124*4882a593Smuzhiyun 
125*4882a593Smuzhiyun 	set_xen_guest_handle(efi_data(op).u.get_variable.name, name);
126*4882a593Smuzhiyun 	BUILD_BUG_ON(sizeof(*vendor) !=
127*4882a593Smuzhiyun 		     sizeof(efi_data(op).u.get_variable.vendor_guid));
128*4882a593Smuzhiyun 	memcpy(&efi_data(op).u.get_variable.vendor_guid, vendor, sizeof(*vendor));
129*4882a593Smuzhiyun 	efi_data(op).u.get_variable.size = *data_size;
130*4882a593Smuzhiyun 	set_xen_guest_handle(efi_data(op).u.get_variable.data, data);
131*4882a593Smuzhiyun 
132*4882a593Smuzhiyun 	if (HYPERVISOR_platform_op(&op) < 0)
133*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
134*4882a593Smuzhiyun 
135*4882a593Smuzhiyun 	*data_size = efi_data(op).u.get_variable.size;
136*4882a593Smuzhiyun 	if (attr)
137*4882a593Smuzhiyun 		*attr = efi_data(op).misc;
138*4882a593Smuzhiyun 
139*4882a593Smuzhiyun 	return efi_data(op).status;
140*4882a593Smuzhiyun }
141*4882a593Smuzhiyun 
xen_efi_get_next_variable(unsigned long * name_size,efi_char16_t * name,efi_guid_t * vendor)142*4882a593Smuzhiyun static efi_status_t xen_efi_get_next_variable(unsigned long *name_size,
143*4882a593Smuzhiyun 					      efi_char16_t *name,
144*4882a593Smuzhiyun 					      efi_guid_t *vendor)
145*4882a593Smuzhiyun {
146*4882a593Smuzhiyun 	struct xen_platform_op op = INIT_EFI_OP(get_next_variable_name);
147*4882a593Smuzhiyun 
148*4882a593Smuzhiyun 	efi_data(op).u.get_next_variable_name.size = *name_size;
149*4882a593Smuzhiyun 	set_xen_guest_handle(efi_data(op).u.get_next_variable_name.name, name);
150*4882a593Smuzhiyun 	BUILD_BUG_ON(sizeof(*vendor) !=
151*4882a593Smuzhiyun 		     sizeof(efi_data(op).u.get_next_variable_name.vendor_guid));
152*4882a593Smuzhiyun 	memcpy(&efi_data(op).u.get_next_variable_name.vendor_guid, vendor,
153*4882a593Smuzhiyun 	       sizeof(*vendor));
154*4882a593Smuzhiyun 
155*4882a593Smuzhiyun 	if (HYPERVISOR_platform_op(&op) < 0)
156*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
157*4882a593Smuzhiyun 
158*4882a593Smuzhiyun 	*name_size = efi_data(op).u.get_next_variable_name.size;
159*4882a593Smuzhiyun 	memcpy(vendor, &efi_data(op).u.get_next_variable_name.vendor_guid,
160*4882a593Smuzhiyun 	       sizeof(*vendor));
161*4882a593Smuzhiyun 
162*4882a593Smuzhiyun 	return efi_data(op).status;
163*4882a593Smuzhiyun }
164*4882a593Smuzhiyun 
xen_efi_set_variable(efi_char16_t * name,efi_guid_t * vendor,u32 attr,unsigned long data_size,void * data)165*4882a593Smuzhiyun static efi_status_t xen_efi_set_variable(efi_char16_t *name, efi_guid_t *vendor,
166*4882a593Smuzhiyun 					 u32 attr, unsigned long data_size,
167*4882a593Smuzhiyun 					 void *data)
168*4882a593Smuzhiyun {
169*4882a593Smuzhiyun 	struct xen_platform_op op = INIT_EFI_OP(set_variable);
170*4882a593Smuzhiyun 
171*4882a593Smuzhiyun 	set_xen_guest_handle(efi_data(op).u.set_variable.name, name);
172*4882a593Smuzhiyun 	efi_data(op).misc = attr;
173*4882a593Smuzhiyun 	BUILD_BUG_ON(sizeof(*vendor) !=
174*4882a593Smuzhiyun 		     sizeof(efi_data(op).u.set_variable.vendor_guid));
175*4882a593Smuzhiyun 	memcpy(&efi_data(op).u.set_variable.vendor_guid, vendor, sizeof(*vendor));
176*4882a593Smuzhiyun 	efi_data(op).u.set_variable.size = data_size;
177*4882a593Smuzhiyun 	set_xen_guest_handle(efi_data(op).u.set_variable.data, data);
178*4882a593Smuzhiyun 
179*4882a593Smuzhiyun 	if (HYPERVISOR_platform_op(&op) < 0)
180*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
181*4882a593Smuzhiyun 
182*4882a593Smuzhiyun 	return efi_data(op).status;
183*4882a593Smuzhiyun }
184*4882a593Smuzhiyun 
xen_efi_query_variable_info(u32 attr,u64 * storage_space,u64 * remaining_space,u64 * max_variable_size)185*4882a593Smuzhiyun static efi_status_t xen_efi_query_variable_info(u32 attr, u64 *storage_space,
186*4882a593Smuzhiyun 						u64 *remaining_space,
187*4882a593Smuzhiyun 						u64 *max_variable_size)
188*4882a593Smuzhiyun {
189*4882a593Smuzhiyun 	struct xen_platform_op op = INIT_EFI_OP(query_variable_info);
190*4882a593Smuzhiyun 
191*4882a593Smuzhiyun 	if (efi.runtime_version < EFI_2_00_SYSTEM_TABLE_REVISION)
192*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
193*4882a593Smuzhiyun 
194*4882a593Smuzhiyun 	efi_data(op).u.query_variable_info.attr = attr;
195*4882a593Smuzhiyun 
196*4882a593Smuzhiyun 	if (HYPERVISOR_platform_op(&op) < 0)
197*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
198*4882a593Smuzhiyun 
199*4882a593Smuzhiyun 	*storage_space = efi_data(op).u.query_variable_info.max_store_size;
200*4882a593Smuzhiyun 	*remaining_space = efi_data(op).u.query_variable_info.remain_store_size;
201*4882a593Smuzhiyun 	*max_variable_size = efi_data(op).u.query_variable_info.max_size;
202*4882a593Smuzhiyun 
203*4882a593Smuzhiyun 	return efi_data(op).status;
204*4882a593Smuzhiyun }
205*4882a593Smuzhiyun 
xen_efi_get_next_high_mono_count(u32 * count)206*4882a593Smuzhiyun static efi_status_t xen_efi_get_next_high_mono_count(u32 *count)
207*4882a593Smuzhiyun {
208*4882a593Smuzhiyun 	struct xen_platform_op op = INIT_EFI_OP(get_next_high_monotonic_count);
209*4882a593Smuzhiyun 
210*4882a593Smuzhiyun 	if (HYPERVISOR_platform_op(&op) < 0)
211*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
212*4882a593Smuzhiyun 
213*4882a593Smuzhiyun 	*count = efi_data(op).misc;
214*4882a593Smuzhiyun 
215*4882a593Smuzhiyun 	return efi_data(op).status;
216*4882a593Smuzhiyun }
217*4882a593Smuzhiyun 
xen_efi_update_capsule(efi_capsule_header_t ** capsules,unsigned long count,unsigned long sg_list)218*4882a593Smuzhiyun static efi_status_t xen_efi_update_capsule(efi_capsule_header_t **capsules,
219*4882a593Smuzhiyun 				unsigned long count, unsigned long sg_list)
220*4882a593Smuzhiyun {
221*4882a593Smuzhiyun 	struct xen_platform_op op = INIT_EFI_OP(update_capsule);
222*4882a593Smuzhiyun 
223*4882a593Smuzhiyun 	if (efi.runtime_version < EFI_2_00_SYSTEM_TABLE_REVISION)
224*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
225*4882a593Smuzhiyun 
226*4882a593Smuzhiyun 	set_xen_guest_handle(efi_data(op).u.update_capsule.capsule_header_array,
227*4882a593Smuzhiyun 			     capsules);
228*4882a593Smuzhiyun 	efi_data(op).u.update_capsule.capsule_count = count;
229*4882a593Smuzhiyun 	efi_data(op).u.update_capsule.sg_list = sg_list;
230*4882a593Smuzhiyun 
231*4882a593Smuzhiyun 	if (HYPERVISOR_platform_op(&op) < 0)
232*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
233*4882a593Smuzhiyun 
234*4882a593Smuzhiyun 	return efi_data(op).status;
235*4882a593Smuzhiyun }
236*4882a593Smuzhiyun 
xen_efi_query_capsule_caps(efi_capsule_header_t ** capsules,unsigned long count,u64 * max_size,int * reset_type)237*4882a593Smuzhiyun static efi_status_t xen_efi_query_capsule_caps(efi_capsule_header_t **capsules,
238*4882a593Smuzhiyun 			unsigned long count, u64 *max_size, int *reset_type)
239*4882a593Smuzhiyun {
240*4882a593Smuzhiyun 	struct xen_platform_op op = INIT_EFI_OP(query_capsule_capabilities);
241*4882a593Smuzhiyun 
242*4882a593Smuzhiyun 	if (efi.runtime_version < EFI_2_00_SYSTEM_TABLE_REVISION)
243*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
244*4882a593Smuzhiyun 
245*4882a593Smuzhiyun 	set_xen_guest_handle(efi_data(op).u.query_capsule_capabilities.capsule_header_array,
246*4882a593Smuzhiyun 					capsules);
247*4882a593Smuzhiyun 	efi_data(op).u.query_capsule_capabilities.capsule_count = count;
248*4882a593Smuzhiyun 
249*4882a593Smuzhiyun 	if (HYPERVISOR_platform_op(&op) < 0)
250*4882a593Smuzhiyun 		return EFI_UNSUPPORTED;
251*4882a593Smuzhiyun 
252*4882a593Smuzhiyun 	*max_size = efi_data(op).u.query_capsule_capabilities.max_capsule_size;
253*4882a593Smuzhiyun 	*reset_type = efi_data(op).u.query_capsule_capabilities.reset_type;
254*4882a593Smuzhiyun 
255*4882a593Smuzhiyun 	return efi_data(op).status;
256*4882a593Smuzhiyun }
257*4882a593Smuzhiyun 
xen_efi_reset_system(int reset_type,efi_status_t status,unsigned long data_size,efi_char16_t * data)258*4882a593Smuzhiyun static void xen_efi_reset_system(int reset_type, efi_status_t status,
259*4882a593Smuzhiyun 				 unsigned long data_size, efi_char16_t *data)
260*4882a593Smuzhiyun {
261*4882a593Smuzhiyun 	switch (reset_type) {
262*4882a593Smuzhiyun 	case EFI_RESET_COLD:
263*4882a593Smuzhiyun 	case EFI_RESET_WARM:
264*4882a593Smuzhiyun 		xen_reboot(SHUTDOWN_reboot);
265*4882a593Smuzhiyun 		break;
266*4882a593Smuzhiyun 	case EFI_RESET_SHUTDOWN:
267*4882a593Smuzhiyun 		xen_reboot(SHUTDOWN_poweroff);
268*4882a593Smuzhiyun 		break;
269*4882a593Smuzhiyun 	default:
270*4882a593Smuzhiyun 		BUG();
271*4882a593Smuzhiyun 	}
272*4882a593Smuzhiyun }
273*4882a593Smuzhiyun 
274*4882a593Smuzhiyun /*
275*4882a593Smuzhiyun  * Set XEN EFI runtime services function pointers. Other fields of struct efi,
276*4882a593Smuzhiyun  * e.g. efi.systab, will be set like normal EFI.
277*4882a593Smuzhiyun  */
xen_efi_runtime_setup(void)278*4882a593Smuzhiyun void __init xen_efi_runtime_setup(void)
279*4882a593Smuzhiyun {
280*4882a593Smuzhiyun 	efi.get_time			= xen_efi_get_time;
281*4882a593Smuzhiyun 	efi.set_time			= xen_efi_set_time;
282*4882a593Smuzhiyun 	efi.get_wakeup_time		= xen_efi_get_wakeup_time;
283*4882a593Smuzhiyun 	efi.set_wakeup_time		= xen_efi_set_wakeup_time;
284*4882a593Smuzhiyun 	efi.get_variable		= xen_efi_get_variable;
285*4882a593Smuzhiyun 	efi.get_next_variable		= xen_efi_get_next_variable;
286*4882a593Smuzhiyun 	efi.set_variable		= xen_efi_set_variable;
287*4882a593Smuzhiyun 	efi.set_variable_nonblocking	= xen_efi_set_variable;
288*4882a593Smuzhiyun 	efi.query_variable_info		= xen_efi_query_variable_info;
289*4882a593Smuzhiyun 	efi.query_variable_info_nonblocking = xen_efi_query_variable_info;
290*4882a593Smuzhiyun 	efi.update_capsule		= xen_efi_update_capsule;
291*4882a593Smuzhiyun 	efi.query_capsule_caps		= xen_efi_query_capsule_caps;
292*4882a593Smuzhiyun 	efi.get_next_high_mono_count	= xen_efi_get_next_high_mono_count;
293*4882a593Smuzhiyun 	efi.reset_system		= xen_efi_reset_system;
294*4882a593Smuzhiyun }
295