[efi] Allow use of EFI configuration tables
[people/asdlkf/gpxe.git] / src / interface / efi / efi_entry.c
1 /*
2  * Copyright (C) 2008 Michael Brown <mbrown@fensystems.co.uk>.
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License as
6  * published by the Free Software Foundation; either version 2 of the
7  * License, or any later version.
8  *
9  * This program is distributed in the hope that it will be useful, but
10  * WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17  */
18
19 #include <stdlib.h>
20 #include <string.h>
21 #include <gpxe/efi/efi.h>
22 #include <gpxe/uuid.h>
23
24 /** Image handle passed to entry point */
25 EFI_HANDLE efi_image_handle;
26
27 /** System table passed to entry point */
28 EFI_SYSTEM_TABLE *efi_systab;
29
30 /** Declared used EFI protocols */
31 static struct efi_protocol efi_protocols[0] \
32         __table_start ( struct efi_protocol, efi_protocols );
33 static struct efi_protocol efi_protocols_end[0] \
34         __table_end ( struct efi_protocol, efi_protocols );
35
36 /** Declared used EFI configuration tables */
37 static struct efi_config_table efi_config_tables[0] \
38         __table_start ( struct efi_config_table, efi_config_tables );
39 static struct efi_config_table efi_config_tables_end[0] \
40         __table_end ( struct efi_config_table, efi_config_tables );
41
42 /**
43  * Look up EFI configuration table
44  *
45  * @v guid              Configuration table GUID
46  * @ret table           Configuration table, or NULL
47  */
48 static void * efi_find_table ( EFI_GUID *guid ) {
49         unsigned int i;
50
51         for ( i = 0 ; i < efi_systab->NumberOfTableEntries ; i++ ) {
52                 if ( memcmp ( &efi_systab->ConfigurationTable[i].VendorGuid,
53                               guid, sizeof ( *guid ) ) == 0 )
54                         return efi_systab->ConfigurationTable[i].VendorTable;
55         }
56
57         return NULL;
58 }
59
60 /**
61  * EFI entry point
62  *
63  * @v image_handle      Image handle
64  * @v systab            System table
65  * @ret efirc           EFI return status code
66  */
67 EFI_STATUS EFIAPI efi_entry ( EFI_HANDLE image_handle,
68                               EFI_SYSTEM_TABLE *systab ) {
69         EFI_BOOT_SERVICES *bs;
70         struct efi_protocol *prot;
71         struct efi_config_table *tab;
72         EFI_STATUS efirc;
73
74         /* Store image handle and system table pointer for future use */
75         efi_image_handle = image_handle;
76         efi_systab = systab;
77
78         /* Sanity checks */
79         if ( ! systab )
80                 return EFI_NOT_AVAILABLE_YET;
81         if ( ! systab->ConOut )
82                 return EFI_NOT_AVAILABLE_YET;
83         if ( ! systab->BootServices ) {
84                 DBGC ( systab, "EFI provided no BootServices entry point\n" );
85                 return EFI_NOT_AVAILABLE_YET;
86         }
87         if ( ! systab->RuntimeServices ) {
88                 DBGC ( systab, "EFI provided no RuntimeServices entry "
89                        "point\n" );
90                 return EFI_NOT_AVAILABLE_YET;
91         }
92         DBGC ( systab, "EFI handle %p systab %p\n", image_handle, systab );
93
94         /* Look up used protocols */
95         bs = systab->BootServices;
96         for ( prot = efi_protocols ; prot < efi_protocols_end ; prot++ ) {
97                 if ( ( efirc = bs->LocateProtocol ( &prot->u.guid, NULL,
98                                                     prot->protocol ) ) == 0 ) {
99                         DBGC ( systab, "EFI protocol %s is at %p\n",
100                                uuid_ntoa ( &prot->u.uuid ), *(prot->protocol));
101                 } else {
102                         DBGC ( systab, "EFI does not provide protocol %s\n",
103                                uuid_ntoa ( &prot->u.uuid ) );
104                         /* All protocols are required */
105                         return efirc;
106                 }
107         }
108
109         /* Look up used configuration tables */
110         for ( tab = efi_config_tables ; tab < efi_config_tables_end ; tab++ ) {
111                 if ( ( *(tab->table) = efi_find_table ( &tab->u.guid ) ) ) {
112                         DBGC ( systab, "EFI configuration table %s is at %p\n",
113                                uuid_ntoa ( &tab->u.uuid ), *(tab->table) );
114                 } else {
115                         DBGC ( systab, "EFI does not provide configuration "
116                                "table %s\n", uuid_ntoa ( &tab->u.uuid ) );
117                         if ( tab->required )
118                                 return EFI_NOT_AVAILABLE_YET;
119                 }
120         }
121
122         /* Call to main() */
123         return RC_TO_EFIRC ( main () );
124 }