b9220f5eaab57ea3ad1304865cf766b9a51e8898
[people/andreif/gpxe.git] / src / net / netdev_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 FILE_LICENCE ( GPL2_OR_LATER );
20
21 #include <string.h>
22 #include <errno.h>
23 #include <gpxe/dhcp.h>
24 #include <gpxe/settings.h>
25 #include <gpxe/netdevice.h>
26
27 /** @file
28  *
29  * Network device configuration settings
30  *
31  */
32
33 /** Network device named settings */
34 struct setting mac_setting __setting = {
35         .name = "mac",
36         .description = "MAC address",
37         .type = &setting_type_hex,
38 };
39
40 /**
41  * Store value of network device setting
42  *
43  * @v settings          Settings block
44  * @v setting           Setting to store
45  * @v data              Setting data, or NULL to clear setting
46  * @v len               Length of setting data
47  * @ret rc              Return status code
48  */
49 static int netdev_store ( struct settings *settings, struct setting *setting,
50                           const void *data, size_t len ) {
51         struct net_device *netdev = container_of ( settings, struct net_device,
52                                                    settings.settings );
53
54         if ( setting_cmp ( setting, &mac_setting ) == 0 ) {
55                 if ( len != netdev->ll_protocol->ll_addr_len )
56                         return -EINVAL;
57                 memcpy ( netdev->ll_addr, data, len );
58                 return 0;
59         }
60
61         return generic_settings_store ( settings, setting, data, len );
62 }
63
64 /**
65  * Fetch value of network device setting
66  *
67  * @v settings          Settings block
68  * @v setting           Setting to fetch
69  * @v data              Setting data, or NULL to clear setting
70  * @v len               Length of setting data
71  * @ret rc              Return status code
72  */
73 static int netdev_fetch ( struct settings *settings, struct setting *setting,
74                           void *data, size_t len ) {
75         struct net_device *netdev = container_of ( settings, struct net_device,
76                                                    settings.settings );
77
78         if ( setting_cmp ( setting, &mac_setting ) == 0 ) {
79                 if ( len > netdev->ll_protocol->ll_addr_len )
80                         len = netdev->ll_protocol->ll_addr_len;
81                 memcpy ( data, netdev->ll_addr, len );
82                 return netdev->ll_protocol->ll_addr_len;
83         }
84
85         return generic_settings_fetch ( settings, setting, data, len );
86 }
87
88 /**
89  * Clear network device settings
90  *
91  * @v settings          Settings block
92  */
93 static void netdev_clear ( struct settings *settings ) {
94         generic_settings_clear ( settings );
95 }
96
97 /** Network device configuration settings operations */
98 struct settings_operations netdev_settings_operations = {
99         .store = netdev_store,
100         .fetch = netdev_fetch,
101         .clear = netdev_clear,
102 };