]> git.ozlabs.org Git - petitboot/blob - discover/platform.c
autotools: Use non-recursive make
[petitboot] / discover / platform.c
1
2 #include <string.h>
3
4 #include <log/log.h>
5 #include <types/types.h>
6 #include <talloc/talloc.h>
7
8 #include "platform.h"
9
10 void                    *platform_ctx;
11 static struct platform  *platform;
12 static struct config    *config;
13
14 static const char *device_type_name(enum device_type type)
15 {
16         switch (type) {
17         case DEVICE_TYPE_DISK:
18                 return "disk";
19         case DEVICE_TYPE_OPTICAL:
20                 return "optical";
21         case DEVICE_TYPE_NETWORK:
22                 return "network";
23         case DEVICE_TYPE_ANY:
24                 return "any";
25         case DEVICE_TYPE_UNKNOWN:
26         default:
27                 return "unknown";
28         }
29 }
30
31 static void dump_config(struct config *config)
32 {
33         unsigned int i;
34
35         pb_log("configuration:\n");
36
37         if (config->autoboot_enabled)
38                 pb_log(" autoboot: enabled, %d sec\n",
39                                 config->autoboot_timeout_sec);
40         else
41                 pb_log(" autoboot: disabled\n");
42
43         if (config->network.n_interfaces || config->network.n_dns_servers)
44                 pb_log(" network configuration:\n");
45
46         if (config->safe_mode)
47                 pb_log(" safe mode: active\n");
48
49         for (i = 0; i < config->network.n_interfaces; i++) {
50                 struct interface_config *ifconf =
51                         config->network.interfaces[i];
52
53                 pb_log("  interface %02x:%02x:%02x:%02x:%02x:%02x\n",
54                                 ifconf->hwaddr[0], ifconf->hwaddr[1],
55                                 ifconf->hwaddr[2], ifconf->hwaddr[3],
56                                 ifconf->hwaddr[4], ifconf->hwaddr[5]);
57
58                 if (ifconf->ignore) {
59                         pb_log("   ignore\n");
60                         continue;
61                 }
62
63                 if (ifconf->method == CONFIG_METHOD_DHCP) {
64                         pb_log("   dhcp\n");
65
66                 } else if (ifconf->method == CONFIG_METHOD_STATIC) {
67                         pb_log("   static:\n");
68                         pb_log("    ip:  %s\n", ifconf->static_config.address);
69                         pb_log("    gw:  %s\n", ifconf->static_config.gateway);
70
71                 }
72         }
73         for (i = 0; i < config->network.n_dns_servers; i++)
74                 pb_log("  dns server %s\n", config->network.dns_servers[i]);
75
76         if (config->n_boot_priorities)
77                 pb_log(" boot priority order:\n");
78
79         for (i = 0; i < config->n_boot_priorities; i++) {
80                 struct boot_priority *prio = &config->boot_priorities[i];
81                 pb_log(" %10s: %d\n", device_type_name(prio->type),
82                                         prio->priority);
83         }
84
85         pb_log(" language: %s\n", config->lang ?: "");
86 }
87
88 void config_set_defaults(struct config *config)
89 {
90         config->autoboot_enabled = true;
91         config->autoboot_timeout_sec = 10;
92         config->autoboot_enabled = true;
93         config->network.interfaces = NULL;
94         config->network.n_interfaces = 0;
95         config->network.dns_servers = NULL;
96         config->network.n_dns_servers = 0;
97         config->boot_device = NULL;
98         config->safe_mode = false;
99         config->lang = NULL;
100
101         config->n_boot_priorities = 2;
102         config->boot_priorities = talloc_array(config, struct boot_priority,
103                                                 config->n_boot_priorities);
104         config->boot_priorities[0].type = DEVICE_TYPE_NETWORK;
105         config->boot_priorities[0].priority = 2;
106         config->boot_priorities[1].type = DEVICE_TYPE_DISK;
107         config->boot_priorities[1].priority = 1;
108 }
109
110 int platform_init(void *ctx)
111 {
112         extern struct platform *__start_platforms,  *__stop_platforms;
113         struct platform **p;
114
115         platform_ctx = talloc_new(ctx);
116
117         for (p = &__start_platforms; p < &__stop_platforms; p++) {
118                 if (!(*p)->probe(*p, platform_ctx))
119                         continue;
120                 platform = *p;
121                 break;
122         }
123
124         config = talloc(platform_ctx, struct config);
125         config_set_defaults(config);
126
127         if (platform) {
128                 pb_log("Detected platform type: %s\n", platform->name);
129                 if (platform->load_config)
130                         platform->load_config(platform, config);
131         } else {
132                 pb_log("No platform type detected, some platform-specific "
133                                 "functionality will be disabled\n");
134         }
135
136         dump_config(config);
137
138         return 0;
139 }
140
141 const struct platform *platform_get(void)
142 {
143         return platform;
144 }
145
146 int config_set(struct config *newconfig)
147 {
148         int rc;
149
150         if (!platform || !platform->save_config)
151                 return -1;
152
153         if (newconfig == config)
154                 return 0;
155
156         pb_log("new configuration data received\n");
157         dump_config(newconfig);
158
159         rc = platform->save_config(platform, newconfig);
160
161         if (!rc)
162                 config = talloc_steal(platform_ctx, newconfig);
163         else
164                 pb_log("error saving new configuration; changes lost\n");
165
166         return rc;
167 }
168
169 /* A non-exported function to allow the test infrastructure to initialise
170  * (and change) the configuration variables */
171 struct parser_test;
172 struct config __attribute__((unused)) *test_config_init(
173                 struct parser_test *test);
174 struct config *test_config_init(struct parser_test *test)
175 {
176         config = talloc(test, struct config);
177         config_set_defaults(config);
178         return config;
179 }
180
181 const struct config *config_get(void)
182 {
183         return config;
184 }
185
186 void config_set_autoboot(bool autoboot_enabled)
187 {
188         config->autoboot_enabled = autoboot_enabled;
189
190         pb_log("set autoboot: %s\n",
191                         config->autoboot_enabled ? "enabled" : "disabled");
192 }
193
194 int platform_fini(void)
195 {
196         talloc_free(platform_ctx);
197         return 0;
198 }