1 /** @file
2
3 Driver for the virtual Xen PCI device
4
5 Copyright (C) 2012, Red Hat, Inc.
6 Copyright (c) 2012 - 2016, Intel Corporation. All rights reserved.<BR>
7 Copyright (C) 2013, ARM Ltd.
8 Copyright (C) 2015, Linaro Ltd.
9
10 This program and the accompanying materials are licensed and made available
11 under the terms and conditions of the BSD License which accompanies this
12 distribution. The full text of the license may be found at
13 http://opensource.org/licenses/bsd-license.php
14
15 THE PROGRAM IS DISTRIBUTED UNDER THE BSD LICENSE ON AN "AS IS" BASIS, WITHOUT
16 WARRANTIES OR REPRESENTATIONS OF ANY KIND, EITHER EXPRESS OR IMPLIED.
17
18 **/
19
20 #include <IndustryStandard/Acpi.h>
21 #include <IndustryStandard/Pci.h>
22 #include <Library/BaseMemoryLib.h>
23 #include <Library/DebugLib.h>
24 #include <Library/MemoryAllocationLib.h>
25 #include <Library/UefiBootServicesTableLib.h>
26 #include <Library/UefiLib.h>
27
28 #include <Protocol/PciIo.h>
29 #include <Protocol/XenIo.h>
30
31 #define PCI_VENDOR_ID_XEN 0x5853
32 #define PCI_DEVICE_ID_XEN_PLATFORM 0x0001
33
34 /**
35
36 Device probe function for this driver.
37
38 The DXE core calls this function for any given device in order to see if the
39 driver can drive the device.
40
41 @param[in] This The EFI_DRIVER_BINDING_PROTOCOL object
42 incorporating this driver (independently of
43 any device).
44
45 @param[in] DeviceHandle The device to probe.
46
47 @param[in] RemainingDevicePath Relevant only for bus drivers, ignored.
48
49
50 @retval EFI_SUCCESS The driver supports the device being probed.
51
52 @retval EFI_UNSUPPORTED The driver does not support the device being probed.
53
54 @return Error codes from the OpenProtocol() boot service or
55 the PciIo protocol.
56
57 **/
58 STATIC
59 EFI_STATUS
60 EFIAPI
XenIoPciDeviceBindingSupported(IN EFI_DRIVER_BINDING_PROTOCOL * This,IN EFI_HANDLE DeviceHandle,IN EFI_DEVICE_PATH_PROTOCOL * RemainingDevicePath)61 XenIoPciDeviceBindingSupported (
62 IN EFI_DRIVER_BINDING_PROTOCOL *This,
63 IN EFI_HANDLE DeviceHandle,
64 IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath
65 )
66 {
67 EFI_STATUS Status;
68 EFI_PCI_IO_PROTOCOL *PciIo;
69 PCI_TYPE00 Pci;
70
71 //
72 // Attempt to open the device with the PciIo set of interfaces. On success,
73 // the protocol is "instantiated" for the PCI device. Covers duplicate open
74 // attempts (EFI_ALREADY_STARTED).
75 //
76 Status = gBS->OpenProtocol (
77 DeviceHandle, // candidate device
78 &gEfiPciIoProtocolGuid, // for generic PCI access
79 (VOID **)&PciIo, // handle to instantiate
80 This->DriverBindingHandle, // requestor driver identity
81 DeviceHandle, // ControllerHandle, according to
82 // the UEFI Driver Model
83 EFI_OPEN_PROTOCOL_BY_DRIVER // get exclusive PciIo access to
84 // the device; to be released
85 );
86 if (EFI_ERROR (Status)) {
87 return Status;
88 }
89
90 //
91 // Read entire PCI configuration header for more extensive check ahead.
92 //
93 Status = PciIo->Pci.Read (
94 PciIo, // (protocol, device)
95 // handle
96 EfiPciIoWidthUint32, // access width & copy
97 // mode
98 0, // Offset
99 sizeof Pci / sizeof (UINT32), // Count
100 &Pci // target buffer
101 );
102
103 if (Status == EFI_SUCCESS) {
104 if ((Pci.Hdr.VendorId == PCI_VENDOR_ID_XEN) &&
105 (Pci.Hdr.DeviceId == PCI_DEVICE_ID_XEN_PLATFORM)) {
106 Status = EFI_SUCCESS;
107 } else {
108 Status = EFI_UNSUPPORTED;
109 }
110 }
111
112 //
113 // We needed PCI IO access only transitorily, to see whether we support the
114 // device or not.
115 //
116 gBS->CloseProtocol (DeviceHandle, &gEfiPciIoProtocolGuid,
117 This->DriverBindingHandle, DeviceHandle);
118
119 return Status;
120 }
121
122 /**
123
124 After we've pronounced support for a specific device in
125 DriverBindingSupported(), we start managing said device (passed in by the
126 Driver Execution Environment) with the following service.
127
128 See DriverBindingSupported() for specification references.
129
130 @param[in] This The EFI_DRIVER_BINDING_PROTOCOL object
131 incorporating this driver (independently of
132 any device).
133
134 @param[in] DeviceHandle The supported device to drive.
135
136 @param[in] RemainingDevicePath Relevant only for bus drivers, ignored.
137
138
139 @retval EFI_SUCCESS The device was started.
140
141 @retval EFI_OUT_OF_RESOURCES Memory allocation failed.
142
143 @return Error codes from the OpenProtocol() boot
144 service, the PciIo protocol or the
145 InstallProtocolInterface() boot service.
146
147 **/
148 STATIC
149 EFI_STATUS
150 EFIAPI
XenIoPciDeviceBindingStart(IN EFI_DRIVER_BINDING_PROTOCOL * This,IN EFI_HANDLE DeviceHandle,IN EFI_DEVICE_PATH_PROTOCOL * RemainingDevicePath)151 XenIoPciDeviceBindingStart (
152 IN EFI_DRIVER_BINDING_PROTOCOL *This,
153 IN EFI_HANDLE DeviceHandle,
154 IN EFI_DEVICE_PATH_PROTOCOL *RemainingDevicePath
155 )
156 {
157 EFI_STATUS Status;
158 XENIO_PROTOCOL *XenIo;
159 EFI_PCI_IO_PROTOCOL *PciIo;
160 EFI_ACPI_ADDRESS_SPACE_DESCRIPTOR *BarDesc;
161
162 XenIo = (XENIO_PROTOCOL *) AllocateZeroPool (sizeof *XenIo);
163 if (XenIo == NULL) {
164 return EFI_OUT_OF_RESOURCES;
165 }
166
167 Status = gBS->OpenProtocol (DeviceHandle, &gEfiPciIoProtocolGuid,
168 (VOID **)&PciIo, This->DriverBindingHandle,
169 DeviceHandle, EFI_OPEN_PROTOCOL_BY_DRIVER);
170 if (EFI_ERROR (Status)) {
171 goto FreeXenIo;
172 }
173
174 //
175 // The BAR1 of this PCI device is used for shared memory and is supposed to
176 // look like MMIO. The address space of the BAR1 will be used to map the
177 // Grant Table.
178 //
179 Status = PciIo->GetBarAttributes (PciIo, PCI_BAR_IDX1, NULL, (VOID**) &BarDesc);
180 ASSERT_EFI_ERROR (Status);
181 ASSERT (BarDesc->ResType == ACPI_ADDRESS_SPACE_TYPE_MEM);
182
183 /* Get a Memory address for mapping the Grant Table. */
184 DEBUG ((EFI_D_INFO, "XenIoPci: BAR at %LX\n", BarDesc->AddrRangeMin));
185 XenIo->GrantTableAddress = BarDesc->AddrRangeMin;
186 FreePool (BarDesc);
187
188 Status = gBS->InstallProtocolInterface (&DeviceHandle,
189 &gXenIoProtocolGuid, EFI_NATIVE_INTERFACE, XenIo);
190
191 if (!EFI_ERROR (Status)) {
192 return EFI_SUCCESS;
193 }
194
195 gBS->CloseProtocol (DeviceHandle, &gEfiPciIoProtocolGuid,
196 This->DriverBindingHandle, DeviceHandle);
197
198 FreeXenIo:
199 FreePool (XenIo);
200
201 return Status;
202 }
203
204 /**
205
206 Stop driving the XenIo PCI device
207
208 @param[in] This The EFI_DRIVER_BINDING_PROTOCOL object
209 incorporating this driver (independently of any
210 device).
211
212 @param[in] DeviceHandle Stop driving this device.
213
214 @param[in] NumberOfChildren Since this function belongs to a device driver
215 only (as opposed to a bus driver), the caller
216 environment sets NumberOfChildren to zero, and
217 we ignore it.
218
219 @param[in] ChildHandleBuffer Ignored (corresponding to NumberOfChildren).
220
221 @retval EFI_SUCCESS Driver instance has been stopped and the PCI
222 configuration attributes have been restored.
223
224 @return Error codes from the OpenProtocol() or
225 CloseProtocol(), UninstallProtocolInterface()
226 boot services.
227
228 **/
229 STATIC
230 EFI_STATUS
231 EFIAPI
XenIoPciDeviceBindingStop(IN EFI_DRIVER_BINDING_PROTOCOL * This,IN EFI_HANDLE DeviceHandle,IN UINTN NumberOfChildren,IN EFI_HANDLE * ChildHandleBuffer)232 XenIoPciDeviceBindingStop (
233 IN EFI_DRIVER_BINDING_PROTOCOL *This,
234 IN EFI_HANDLE DeviceHandle,
235 IN UINTN NumberOfChildren,
236 IN EFI_HANDLE *ChildHandleBuffer
237 )
238 {
239 EFI_STATUS Status;
240 XENIO_PROTOCOL *XenIo;
241
242 Status = gBS->OpenProtocol (
243 DeviceHandle, // candidate device
244 &gXenIoProtocolGuid, // retrieve the XenIo iface
245 (VOID **)&XenIo, // target pointer
246 This->DriverBindingHandle, // requestor driver identity
247 DeviceHandle, // requesting lookup for dev.
248 EFI_OPEN_PROTOCOL_GET_PROTOCOL // lookup only, no ref. added
249 );
250 if (EFI_ERROR (Status)) {
251 return Status;
252 }
253
254 //
255 // Handle Stop() requests for in-use driver instances gracefully.
256 //
257 Status = gBS->UninstallProtocolInterface (DeviceHandle,
258 &gXenIoProtocolGuid, XenIo);
259 if (EFI_ERROR (Status)) {
260 return Status;
261 }
262
263 Status = gBS->CloseProtocol (DeviceHandle, &gEfiPciIoProtocolGuid,
264 This->DriverBindingHandle, DeviceHandle);
265
266 FreePool (XenIo);
267
268 return Status;
269 }
270
271
272 //
273 // The static object that groups the Supported() (ie. probe), Start() and
274 // Stop() functions of the driver together. Refer to UEFI Spec 2.3.1 + Errata
275 // C, 10.1 EFI Driver Binding Protocol.
276 //
277 STATIC EFI_DRIVER_BINDING_PROTOCOL gDriverBinding = {
278 &XenIoPciDeviceBindingSupported,
279 &XenIoPciDeviceBindingStart,
280 &XenIoPciDeviceBindingStop,
281 0x10, // Version, must be in [0x10 .. 0xFFFFFFEF] for IHV-developed drivers
282 NULL, // ImageHandle, to be overwritten by
283 // EfiLibInstallDriverBindingComponentName2() in XenIoPciDeviceEntryPoint()
284 NULL // DriverBindingHandle, ditto
285 };
286
287
288 //
289 // The purpose of the following scaffolding (EFI_COMPONENT_NAME_PROTOCOL and
290 // EFI_COMPONENT_NAME2_PROTOCOL implementation) is to format the driver's name
291 // in English, for display on standard console devices. This is recommended for
292 // UEFI drivers that follow the UEFI Driver Model. Refer to the Driver Writer's
293 // Guide for UEFI 2.3.1 v1.01, 11 UEFI Driver and Controller Names.
294 //
295 STATIC
296 EFI_UNICODE_STRING_TABLE mDriverNameTable[] = {
297 { "eng;en", L"XenIo PCI Driver" },
298 { NULL, NULL }
299 };
300
301 STATIC
302 EFI_COMPONENT_NAME_PROTOCOL gComponentName;
303
304 EFI_STATUS
305 EFIAPI
XenIoPciGetDriverName(IN EFI_COMPONENT_NAME_PROTOCOL * This,IN CHAR8 * Language,OUT CHAR16 ** DriverName)306 XenIoPciGetDriverName (
307 IN EFI_COMPONENT_NAME_PROTOCOL *This,
308 IN CHAR8 *Language,
309 OUT CHAR16 **DriverName
310 )
311 {
312 return LookupUnicodeString2 (
313 Language,
314 This->SupportedLanguages,
315 mDriverNameTable,
316 DriverName,
317 (BOOLEAN)(This == &gComponentName) // Iso639Language
318 );
319 }
320
321 EFI_STATUS
322 EFIAPI
XenIoPciGetDeviceName(IN EFI_COMPONENT_NAME_PROTOCOL * This,IN EFI_HANDLE DeviceHandle,IN EFI_HANDLE ChildHandle,IN CHAR8 * Language,OUT CHAR16 ** ControllerName)323 XenIoPciGetDeviceName (
324 IN EFI_COMPONENT_NAME_PROTOCOL *This,
325 IN EFI_HANDLE DeviceHandle,
326 IN EFI_HANDLE ChildHandle,
327 IN CHAR8 *Language,
328 OUT CHAR16 **ControllerName
329 )
330 {
331 return EFI_UNSUPPORTED;
332 }
333
334 STATIC
335 EFI_COMPONENT_NAME_PROTOCOL gComponentName = {
336 &XenIoPciGetDriverName,
337 &XenIoPciGetDeviceName,
338 "eng" // SupportedLanguages, ISO 639-2 language codes
339 };
340
341 STATIC
342 EFI_COMPONENT_NAME2_PROTOCOL gComponentName2 = {
343 (EFI_COMPONENT_NAME2_GET_DRIVER_NAME) &XenIoPciGetDriverName,
344 (EFI_COMPONENT_NAME2_GET_CONTROLLER_NAME) &XenIoPciGetDeviceName,
345 "en" // SupportedLanguages, RFC 4646 language codes
346 };
347
348
349 //
350 // Entry point of this driver.
351 //
352 EFI_STATUS
353 EFIAPI
XenIoPciDeviceEntryPoint(IN EFI_HANDLE ImageHandle,IN EFI_SYSTEM_TABLE * SystemTable)354 XenIoPciDeviceEntryPoint (
355 IN EFI_HANDLE ImageHandle,
356 IN EFI_SYSTEM_TABLE *SystemTable
357 )
358 {
359 return EfiLibInstallDriverBindingComponentName2 (
360 ImageHandle,
361 SystemTable,
362 &gDriverBinding,
363 ImageHandle,
364 &gComponentName,
365 &gComponentName2
366 );
367 }
368