blob: 25b27ece6dd4294e484b36a601b2bc059cca5dd4 [file] [log] [blame]
Tom Rinif739fcd2018-05-07 17:02:21 -04001// SPDX-License-Identifier: GPL-2.0+
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +01002/*
3 * Uclass for EFI drivers
4 *
5 * Copyright (c) 2017 Heinrich Schuchardt
6 *
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +01007 * For each EFI driver the uclass
8 * - creates a handle
9 * - installs the driver binding protocol
10 *
11 * The uclass provides the bind, start, and stop entry points for the driver
12 * binding protocol.
13 *
14 * In bind() and stop() it checks if the controller implements the protocol
15 * supported by the EFI driver. In the start() function it calls the bind()
16 * function of the EFI driver. In the stop() function it destroys the child
17 * controllers.
18 */
19
20#include <efi_driver.h>
21
Heinrich Schuchardtea808852018-09-18 18:52:46 +020022/**
23 * check_node_type() - check node type
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +010024 *
Heinrich Schuchardtea808852018-09-18 18:52:46 +020025 * We do not support partitions as controller handles.
26 *
27 * @handle: handle to be checked
28 * Return: status code
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +010029 */
30static efi_status_t check_node_type(efi_handle_t handle)
31{
32 efi_status_t r, ret = EFI_SUCCESS;
33 const struct efi_device_path *dp;
34
35 /* Open the device path protocol */
36 r = EFI_CALL(systab.boottime->open_protocol(
37 handle, &efi_guid_device_path, (void **)&dp,
38 NULL, NULL, EFI_OPEN_PROTOCOL_GET_PROTOCOL));
39 if (r == EFI_SUCCESS && dp) {
40 /* Get the last node */
41 const struct efi_device_path *node = efi_dp_last_node(dp);
42 /* We do not support partitions as controller */
43 if (!node || node->type == DEVICE_PATH_TYPE_MEDIA_DEVICE)
44 ret = EFI_UNSUPPORTED;
45 }
46 return ret;
47}
48
Heinrich Schuchardtea808852018-09-18 18:52:46 +020049/**
50 * efi_uc_supported() - check if the driver supports the controller
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +010051 *
Heinrich Schuchardtea808852018-09-18 18:52:46 +020052 * @this: driver binding protocol
53 * @controller_handle: handle of the controller
54 * @remaining_device_path: path specifying the child controller
55 * Return: status code
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +010056 */
57static efi_status_t EFIAPI efi_uc_supported(
58 struct efi_driver_binding_protocol *this,
59 efi_handle_t controller_handle,
60 struct efi_device_path *remaining_device_path)
61{
62 efi_status_t r, ret;
63 void *interface;
64 struct efi_driver_binding_extended_protocol *bp =
65 (struct efi_driver_binding_extended_protocol *)this;
66
67 EFI_ENTRY("%p, %p, %ls", this, controller_handle,
68 efi_dp_str(remaining_device_path));
69
70 ret = EFI_CALL(systab.boottime->open_protocol(
71 controller_handle, bp->ops->protocol,
72 &interface, this->driver_binding_handle,
73 controller_handle, EFI_OPEN_PROTOCOL_BY_DRIVER));
74 switch (ret) {
75 case EFI_ACCESS_DENIED:
76 case EFI_ALREADY_STARTED:
77 goto out;
78 case EFI_SUCCESS:
79 break;
80 default:
81 ret = EFI_UNSUPPORTED;
82 goto out;
83 }
84
85 ret = check_node_type(controller_handle);
86
87 r = EFI_CALL(systab.boottime->close_protocol(
88 controller_handle, bp->ops->protocol,
89 this->driver_binding_handle,
90 controller_handle));
91 if (r != EFI_SUCCESS)
92 ret = EFI_UNSUPPORTED;
93out:
94 return EFI_EXIT(ret);
95}
96
Heinrich Schuchardtea808852018-09-18 18:52:46 +020097/**
98 * efi_uc_start() - create child controllers and attach driver
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +010099 *
Heinrich Schuchardtea808852018-09-18 18:52:46 +0200100 * @this: driver binding protocol
101 * @controller_handle: handle of the controller
102 * @remaining_device_path: path specifying the child controller
103 * Return: status code
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100104 */
105static efi_status_t EFIAPI efi_uc_start(
106 struct efi_driver_binding_protocol *this,
107 efi_handle_t controller_handle,
108 struct efi_device_path *remaining_device_path)
109{
110 efi_status_t r, ret;
111 void *interface = NULL;
112 struct efi_driver_binding_extended_protocol *bp =
113 (struct efi_driver_binding_extended_protocol *)this;
114
Heinrich Schuchardt30ed1d42020-01-10 12:33:16 +0100115 EFI_ENTRY("%p, %p, %ls", this, controller_handle,
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100116 efi_dp_str(remaining_device_path));
117
118 /* Attach driver to controller */
119 ret = EFI_CALL(systab.boottime->open_protocol(
120 controller_handle, bp->ops->protocol,
121 &interface, this->driver_binding_handle,
122 controller_handle, EFI_OPEN_PROTOCOL_BY_DRIVER));
123 switch (ret) {
124 case EFI_ACCESS_DENIED:
125 case EFI_ALREADY_STARTED:
126 goto out;
127 case EFI_SUCCESS:
128 break;
129 default:
130 ret = EFI_UNSUPPORTED;
131 goto out;
132 }
133 ret = check_node_type(controller_handle);
134 if (ret != EFI_SUCCESS) {
135 r = EFI_CALL(systab.boottime->close_protocol(
136 controller_handle, bp->ops->protocol,
137 this->driver_binding_handle,
138 controller_handle));
139 if (r != EFI_SUCCESS)
140 EFI_PRINT("Failure to close handle\n");
141 goto out;
142 }
143
144 /* TODO: driver specific stuff */
145 bp->ops->bind(controller_handle, interface);
146
147out:
148 return EFI_EXIT(ret);
149}
150
Heinrich Schuchardtea808852018-09-18 18:52:46 +0200151/**
152 * disconnect_child() - remove a single child controller from the parent
153 * controller
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100154 *
Heinrich Schuchardtea808852018-09-18 18:52:46 +0200155 * @controller_handle: parent controller
156 * @child_handle: child controller
157 * Return: status code
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100158 */
159static efi_status_t disconnect_child(efi_handle_t controller_handle,
160 efi_handle_t child_handle)
161{
162 efi_status_t ret;
163 efi_guid_t *guid_controller = NULL;
164 efi_guid_t *guid_child_controller = NULL;
165
166 ret = EFI_CALL(systab.boottime->close_protocol(
167 controller_handle, guid_controller,
168 child_handle, child_handle));
169 if (ret != EFI_SUCCESS) {
170 EFI_PRINT("Cannot close protocol\n");
171 return ret;
172 }
173 ret = EFI_CALL(systab.boottime->uninstall_protocol_interface(
174 child_handle, guid_child_controller, NULL));
175 if (ret != EFI_SUCCESS) {
176 EFI_PRINT("Cannot uninstall protocol interface\n");
177 return ret;
178 }
179 return ret;
180}
181
Heinrich Schuchardtea808852018-09-18 18:52:46 +0200182/**
183 * efi_uc_stop() - Remove child controllers and disconnect the controller
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100184 *
Heinrich Schuchardtea808852018-09-18 18:52:46 +0200185 * @this: driver binding protocol
186 * @controller_handle: handle of the controller
187 * @number_of_children: number of child controllers to remove
188 * @child_handle_buffer: handles of the child controllers to remove
189 * Return: status code
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100190 */
191static efi_status_t EFIAPI efi_uc_stop(
192 struct efi_driver_binding_protocol *this,
193 efi_handle_t controller_handle,
194 size_t number_of_children,
195 efi_handle_t *child_handle_buffer)
196{
197 efi_status_t ret;
198 efi_uintn_t count;
199 struct efi_open_protocol_info_entry *entry_buffer;
Heinrich Schuchardtd7431042020-01-09 23:26:43 +0100200 struct efi_driver_binding_extended_protocol *bp =
201 (struct efi_driver_binding_extended_protocol *)this;
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100202
Heinrich Schuchardt30ed1d42020-01-10 12:33:16 +0100203 EFI_ENTRY("%p, %p, %zu, %p", this, controller_handle,
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100204 number_of_children, child_handle_buffer);
205
206 /* Destroy provided child controllers */
207 if (number_of_children) {
208 efi_uintn_t i;
209
210 for (i = 0; i < number_of_children; ++i) {
211 ret = disconnect_child(controller_handle,
212 child_handle_buffer[i]);
213 if (ret != EFI_SUCCESS)
214 return ret;
215 }
216 return EFI_SUCCESS;
217 }
218
219 /* Destroy all children */
220 ret = EFI_CALL(systab.boottime->open_protocol_information(
Heinrich Schuchardtd7431042020-01-09 23:26:43 +0100221 controller_handle, bp->ops->protocol,
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100222 &entry_buffer, &count));
223 if (ret != EFI_SUCCESS)
224 goto out;
225 while (count) {
226 if (entry_buffer[--count].attributes &
227 EFI_OPEN_PROTOCOL_BY_CHILD_CONTROLLER) {
228 ret = disconnect_child(
229 controller_handle,
230 entry_buffer[count].agent_handle);
231 if (ret != EFI_SUCCESS)
232 goto out;
233 }
234 }
235 ret = EFI_CALL(systab.boottime->free_pool(entry_buffer));
236 if (ret != EFI_SUCCESS)
Heinrich Schuchardtb4f471f2019-01-06 17:12:43 +0100237 printf("%s: ERROR: Cannot free pool\n", __func__);
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100238
239 /* Detach driver from controller */
240 ret = EFI_CALL(systab.boottime->close_protocol(
Heinrich Schuchardtd7431042020-01-09 23:26:43 +0100241 controller_handle, bp->ops->protocol,
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100242 this->driver_binding_handle, controller_handle));
243out:
244 return EFI_EXIT(ret);
245}
246
Heinrich Schuchardtea808852018-09-18 18:52:46 +0200247/**
248 * efi_add_driver() - add driver
249 *
250 * @drv: driver to add
251 * Return: status code
252 */
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100253static efi_status_t efi_add_driver(struct driver *drv)
254{
255 efi_status_t ret;
256 const struct efi_driver_ops *ops = drv->ops;
257 struct efi_driver_binding_extended_protocol *bp;
258
259 debug("EFI: Adding driver '%s'\n", drv->name);
260 if (!ops->protocol) {
261 printf("EFI: ERROR: protocol GUID missing for driver '%s'\n",
262 drv->name);
263 return EFI_INVALID_PARAMETER;
264 }
265 bp = calloc(1, sizeof(struct efi_driver_binding_extended_protocol));
266 if (!bp)
267 return EFI_OUT_OF_RESOURCES;
268
269 bp->bp.supported = efi_uc_supported;
270 bp->bp.start = efi_uc_start;
271 bp->bp.stop = efi_uc_stop;
272 bp->bp.version = 0xffffffff;
273 bp->ops = drv->ops;
274
275 ret = efi_create_handle(&bp->bp.driver_binding_handle);
276 if (ret != EFI_SUCCESS) {
277 free(bp);
278 goto out;
279 }
280 bp->bp.image_handle = bp->bp.driver_binding_handle;
281 ret = efi_add_protocol(bp->bp.driver_binding_handle,
282 &efi_guid_driver_binding_protocol, bp);
283 if (ret != EFI_SUCCESS) {
284 efi_delete_handle(bp->bp.driver_binding_handle);
285 free(bp);
286 goto out;
287 }
288out:
289 return ret;
290}
291
Heinrich Schuchardtea808852018-09-18 18:52:46 +0200292/**
293 * efi_driver_init() - initialize the EFI drivers
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100294 *
Heinrich Schuchardtea808852018-09-18 18:52:46 +0200295 * Called by efi_init_obj_list().
296 *
297 * Return: 0 = success, any other value will stop further execution
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100298 */
Heinrich Schuchardt038782a2018-02-01 12:53:32 +0100299efi_status_t efi_driver_init(void)
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100300{
301 struct driver *drv;
Heinrich Schuchardt038782a2018-02-01 12:53:32 +0100302 efi_status_t ret = EFI_SUCCESS;
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100303
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100304 debug("EFI: Initializing EFI driver framework\n");
305 for (drv = ll_entry_start(struct driver, driver);
306 drv < ll_entry_end(struct driver, driver); ++drv) {
307 if (drv->id == UCLASS_EFI) {
308 ret = efi_add_driver(drv);
Heinrich Schuchardt038782a2018-02-01 12:53:32 +0100309 if (ret != EFI_SUCCESS) {
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100310 printf("EFI: ERROR: failed to add driver %s\n",
311 drv->name);
312 break;
313 }
314 }
315 }
316 return ret;
317}
318
Heinrich Schuchardtea808852018-09-18 18:52:46 +0200319/**
320 * efi_uc_init() - initialize the EFI uclass
321 *
322 * @class: the EFI uclass
323 * Return: 0 = success
324 */
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100325static int efi_uc_init(struct uclass *class)
326{
327 printf("EFI: Initializing UCLASS_EFI\n");
328 return 0;
329}
330
Heinrich Schuchardtea808852018-09-18 18:52:46 +0200331/**
332 * efi_uc_destroy() - destroy the EFI uclass
333 *
334 * @class: the EFI uclass
335 * Return: 0 = success
336 */
Heinrich Schuchardt05ef48a2018-01-21 19:29:30 +0100337static int efi_uc_destroy(struct uclass *class)
338{
339 printf("Destroying UCLASS_EFI\n");
340 return 0;
341}
342
343UCLASS_DRIVER(efi) = {
344 .name = "efi",
345 .id = UCLASS_EFI,
346 .init = efi_uc_init,
347 .destroy = efi_uc_destroy,
348};