[Settings] Expose SMBIOS via settings API
[people/mcb30/gpxe.git] / src / arch / i386 / firmware / pcbios / smbios_settings.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 <stdint.h>
20 #include <string.h>
21 #include <errno.h>
22 #include <gpxe/settings.h>
23 #include <gpxe/init.h>
24 #include <gpxe/uuid.h>
25 #include <smbios.h>
26
27 /**
28  * Construct SMBIOS raw-data tag
29  *
30  * @v _type             SMBIOS structure type number
31  * @v _structure        SMBIOS structure data type
32  * @v _field            Field within SMBIOS structure data type
33  * @ret tag             SMBIOS setting tag
34  */
35 #define SMBIOS_RAW_TAG( _type, _structure, _field )             \
36         ( ( (_type) << 16 ) |                                   \
37           ( offsetof ( _structure, _field ) << 8 ) |            \
38           ( sizeof ( ( ( _structure * ) 0 )->_field ) ) )
39
40 /**
41  * Construct SMBIOS string tag
42  *
43  * @v _type             SMBIOS structure type number
44  * @v _structure        SMBIOS structure data type
45  * @v _field            Field within SMBIOS structure data type
46  * @ret tag             SMBIOS setting tag
47  */
48 #define SMBIOS_STRING_TAG( _type, _structure, _field )          \
49         ( ( (_type) << 16 ) |                                   \
50           ( offsetof ( _structure, _field ) << 8 ) )
51
52 /**
53  * Store value of SMBIOS setting
54  *
55  * @v settings          Settings block
56  * @v setting           Setting to store
57  * @v data              Setting data, or NULL to clear setting
58  * @v len               Length of setting data
59  * @ret rc              Return status code
60  */
61 static int smbios_store ( struct settings *settings __unused,
62                           struct setting *setting __unused,
63                           const void *data __unused, size_t len __unused ) {
64         /* Cannot write data into SMBIOS */
65         return -ENOTSUP;
66 }
67
68 /**
69  * Fetch value of SMBIOS setting
70  *
71  * @v settings          Settings block, or NULL to search all blocks
72  * @v setting           Setting to fetch
73  * @v data              Buffer to fill with setting data
74  * @v len               Length of buffer
75  * @ret len             Length of setting data, or negative error
76  */
77 static int smbios_fetch ( struct settings *settings __unused,
78                           struct setting *setting,
79                           void *data, size_t len ) {
80         struct smbios_structure structure;
81         unsigned int tag_type;
82         unsigned int tag_offset;
83         unsigned int tag_len;
84         int rc;
85
86         /* Split tag into type, offset and length */
87         tag_type = ( setting->tag >> 16 );
88         tag_offset = ( ( setting->tag >> 8 ) & 0xff );
89         tag_len = ( setting->tag & 0xff );
90         if ( ! tag_type )
91                 return -ENOENT;
92
93         /* Find SMBIOS structure */
94         if ( ( rc = find_smbios_structure ( tag_type, &structure ) ) != 0 )
95                 return rc;
96
97         {
98                 uint8_t buf[structure.header.len];
99
100                 /* Read SMBIOS structure */
101                 if ( ( rc = read_smbios_structure ( &structure, buf,
102                                                     sizeof ( buf ) ) ) != 0 )
103                         return rc;
104
105                 if ( tag_len == 0 ) {
106                         /* String */
107                         return read_smbios_string ( &structure,
108                                                     buf[tag_offset],
109                                                     data, len );
110                 } else {
111                         /* Raw data */
112                         if ( len > tag_len )
113                                 len = tag_len;
114                         memcpy ( data, &buf[tag_offset], len );
115                         return tag_len;
116                 }
117         }
118 }
119
120 /** SMBIOS settings operations */
121 static struct settings_operations smbios_settings_operations = {
122         .store = smbios_store,
123         .fetch = smbios_fetch,
124 };
125
126 /** SMBIOS settings */
127 static struct settings smbios_settings = {
128         .refcnt = NULL,
129         .name = "smbios",
130         .siblings = LIST_HEAD_INIT ( smbios_settings.siblings ),
131         .children = LIST_HEAD_INIT ( smbios_settings.children ),
132         .op = &smbios_settings_operations,
133 };
134
135 /** Initialise SMBIOS settings */
136 static void smbios_init ( void ) {
137         int rc;
138
139         if ( ( rc = register_settings ( &smbios_settings, NULL ) ) != 0 ) {
140                 DBG ( "SMBIOS could not register settings: %s\n",
141                       strerror ( rc ) );
142                 return;
143         }
144 }
145
146 /** SMBIOS settings initialiser */
147 struct init_fn smbios_init_fn __init_fn ( INIT_NORMAL ) = {
148         .initialise = smbios_init,
149 };
150
151 /** UUID setting obtained via SMBIOS */
152 struct setting uuid_setting __setting = {
153         .name = "uuid",
154         .description = "UUID",
155         .tag = SMBIOS_RAW_TAG ( SMBIOS_TYPE_SYSTEM_INFORMATION,
156                                 struct smbios_system_information, uuid ),
157         .type = &setting_type_uuid,
158 };