iPXE
bios_smbios.c
Go to the documentation of this file.
1 /*
2  * Copyright (C) 2007 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., 51 Franklin Street, Fifth Floor, Boston, MA
17  * 02110-1301, USA.
18  *
19  * You can also choose to distribute this program under the terms of
20  * the Unmodified Binary Distribution Licence (as given in the file
21  * COPYING.UBDL), provided that you have satisfied its requirements.
22  */
23 
24 FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
25 
26 #include <stdint.h>
27 #include <string.h>
28 #include <errno.h>
29 #include <assert.h>
30 #include <ipxe/uaccess.h>
31 #include <ipxe/smbios.h>
32 #include <realmode.h>
33 #include <pnpbios.h>
34 
35 /** @file
36  *
37  * System Management BIOS
38  *
39  */
40 
41 /**
42  * Find SMBIOS
43  *
44  * @v smbios SMBIOS entry point descriptor structure to fill in
45  * @ret rc Return status code
46  */
47 static int bios_find_smbios2 ( struct smbios *smbios ) {
48  struct smbios_entry entry;
49  int rc;
50 
51  /* Scan through BIOS segment to find SMBIOS 32-bit entry point */
52  if ( ( rc = find_smbios_entry ( real_to_user ( BIOS_SEG, 0 ), 0x10000,
53  &entry ) ) != 0 )
54  return rc;
55 
56  /* Fill in entry point descriptor structure */
57  smbios->address = phys_to_user ( entry.smbios_address );
58  smbios->len = entry.smbios_len;
59  smbios->count = entry.smbios_count;
60  smbios->version = SMBIOS_VERSION ( entry.major, entry.minor );
61 
62  return 0;
63 }
64 
65 /**
66  * Find SMBIOS
67  *
68  * @v smbios SMBIOS entry point descriptor structure to fill in
69  * @ret rc Return status code
70  */
71 static int bios_find_smbios3 ( struct smbios *smbios ) {
72  struct smbios3_entry entry;
73  int rc;
74 
75  /* Scan through BIOS segment to find SMBIOS 64-bit entry point */
76  if ( ( rc = find_smbios3_entry ( real_to_user ( BIOS_SEG, 0 ), 0x10000,
77  &entry ) ) != 0 )
78  return rc;
79 
80  /* Check that address is accessible */
81  if ( entry.smbios_address > ~( ( physaddr_t ) 0 ) ) {
82  DBG ( "SMBIOS3 at %08llx is inaccessible\n",
83  ( ( unsigned long long ) entry.smbios_address ) );
84  return -ENOTSUP;
85  }
86 
87  /* Fill in entry point descriptor structure */
88  smbios->address = phys_to_user ( entry.smbios_address );
89  smbios->len = entry.smbios_len;
90  smbios->count = 0;
91  smbios->version = SMBIOS_VERSION ( entry.major, entry.minor );
92 
93  return 0;
94 }
95 
96 /**
97  * Find SMBIOS
98  *
99  * @v smbios SMBIOS entry point descriptor structure to fill in
100  * @ret rc Return status code
101  */
102 static int bios_find_smbios ( struct smbios *smbios ) {
103  int rc;
104 
105  /* Use 32-bit table if present */
106  if ( ( rc = bios_find_smbios2 ( smbios ) ) == 0 )
107  return 0;
108 
109  /* Otherwise, use 64-bit table if present and accessible */
110  if ( ( rc = bios_find_smbios3 ( smbios ) ) == 0 )
111  return 0;
112 
113  return rc;
114 }
115 
struct arbelprm_rc_send_wqe rc
Definition: arbel.h:14
unsigned int count
Number of SMBIOS structures.
Definition: smbios.h:213
Error codes.
size_t len
Length of SMBIOS structures.
Definition: smbios.h:211
userptr_t phys_to_user(unsigned long phys_addr)
Convert physical address to user pointer.
SMBIOS entry point descriptor.
Definition: smbios.h:207
int find_smbios(struct smbios *smbios)
Access to external ("user") memory.
uint16_t version
SMBIOS version.
Definition: smbios.h:215
#define ENOTSUP
Operation not supported.
Definition: errno.h:589
static int bios_find_smbios2(struct smbios *smbios)
Find SMBIOS.
Definition: bios_smbios.c:47
PROVIDE_SMBIOS(pcbios, find_smbios, bios_find_smbios)
Assertions.
int find_smbios_entry(userptr_t start, size_t len, struct smbios_entry *entry)
Scan for SMBIOS 32-bit entry point structure.
Definition: smbios.c:72
SMBIOS 32-bit entry point.
Definition: smbios.h:49
#define SMBIOS_VERSION(major, minor)
Calculate SMBIOS version.
Definition: smbios.h:225
union aes_table_entry entry[256]
Table entries, indexed by S(N)
Definition: aes.c:26
PnP BIOS.
System Management BIOS.
unsigned long physaddr_t
Definition: stdint.h:20
int find_smbios3_entry(userptr_t start, size_t len, struct smbios3_entry *entry)
Scan for SMBIOS 64-bit entry point structure.
Definition: smbios.c:112
static int bios_find_smbios(struct smbios *smbios)
Find SMBIOS.
Definition: bios_smbios.c:102
userptr_t address
Start of SMBIOS structures.
Definition: smbios.h:209
static int bios_find_smbios3(struct smbios *smbios)
Find SMBIOS.
Definition: bios_smbios.c:71
static __always_inline userptr_t real_to_user(unsigned int segment, unsigned int offset)
Convert segment:offset address to user buffer.
Definition: realmode.h:75
FILE_LICENCE(GPL2_OR_LATER_OR_UBDL)
#define DBG(...)
Print a debugging message.
Definition: compiler.h:498
#define BIOS_SEG
Definition: pnpbios.h:13
String functions.
SMBIOS 64-bit entry point.
Definition: smbios.h:90