aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJeff Dike <jdike@addtoit.com>2006-09-29 04:58:46 -0400
committerLinus Torvalds <torvalds@g5.osdl.org>2006-09-29 12:18:04 -0400
commitf3e7ed2b617824f79d1223f37430ccffae59e5b8 (patch)
tree629961c5ea7aa34675c8a6760e88e3957099f0af
parent3b08606dc2991bcdab14139efd9ed9d492f5f901 (diff)
[PATCH] uml: assign random MACs to interfaces if necessary
Assign a random MAC to an ethernet interface if one was not provided on the command line. This became pressing when distros started bringing interfaces up before assigning IPs to them. The previous pattern of assigning an IP then bringing it up allowed the MAC to be generated from the first IP assigned. However, once the thing is up, it's probably a bad idea to change the MAC, so the MAC stayed initialized to fe:fd:0:0:0:0. Now, if there is no MAC from the command line, one is generated. We use the microseconds from gettimeofday (20 bits), plus the low 12 bits of the pid to seed the random number generator. random() is called twice, with 16 bits of each result used. I didn't want to have to try to fill in 32 bits optimally given an arbitrary RAND_MAX, so I just assume that it is greater than 65536 and use 16 bits of each random() return. There is also a bit of reformatting and whitespace cleanup here. Signed-off-by: Jeff Dike <jdike@addtoit.com> Signed-off-by: Andrew Morton <akpm@osdl.org> Signed-off-by: Linus Torvalds <torvalds@osdl.org>
-rw-r--r--arch/um/drivers/net_kern.c15
-rw-r--r--arch/um/drivers/net_user.c30
-rw-r--r--arch/um/include/net_user.h19
3 files changed, 46 insertions, 18 deletions
diff --git a/arch/um/drivers/net_kern.c b/arch/um/drivers/net_kern.c
index bd1178fa4e9a..684a1ef93c87 100644
--- a/arch/um/drivers/net_kern.c
+++ b/arch/um/drivers/net_kern.c
@@ -753,7 +753,8 @@ int setup_etheraddr(char *str, unsigned char *addr)
753 int i; 753 int i;
754 754
755 if(str == NULL) 755 if(str == NULL)
756 return(0); 756 goto random;
757
757 for(i=0;i<6;i++){ 758 for(i=0;i<6;i++){
758 addr[i] = simple_strtoul(str, &end, 16); 759 addr[i] = simple_strtoul(str, &end, 16);
759 if((end == str) || 760 if((end == str) ||
@@ -761,7 +762,7 @@ int setup_etheraddr(char *str, unsigned char *addr)
761 printk(KERN_ERR 762 printk(KERN_ERR
762 "setup_etheraddr: failed to parse '%s' " 763 "setup_etheraddr: failed to parse '%s' "
763 "as an ethernet address\n", str); 764 "as an ethernet address\n", str);
764 return(0); 765 goto random;
765 } 766 }
766 str = end + 1; 767 str = end + 1;
767 } 768 }
@@ -769,9 +770,15 @@ int setup_etheraddr(char *str, unsigned char *addr)
769 printk(KERN_ERR 770 printk(KERN_ERR
770 "Attempt to assign a broadcast ethernet address to a " 771 "Attempt to assign a broadcast ethernet address to a "
771 "device disallowed\n"); 772 "device disallowed\n");
772 return(0); 773 goto random;
773 } 774 }
774 return(1); 775 return 1;
776
777random:
778 addr[0] = 0xfe;
779 addr[1] = 0xfd;
780 random_mac(addr);
781 return 1;
775} 782}
776 783
777void dev_ip_addr(void *d, unsigned char *bin_buf) 784void dev_ip_addr(void *d, unsigned char *bin_buf)
diff --git a/arch/um/drivers/net_user.c b/arch/um/drivers/net_user.c
index 107c5e43fa00..142bcb2c7c6a 100644
--- a/arch/um/drivers/net_user.c
+++ b/arch/um/drivers/net_user.c
@@ -12,6 +12,7 @@
12#include <string.h> 12#include <string.h>
13#include <sys/socket.h> 13#include <sys/socket.h>
14#include <sys/wait.h> 14#include <sys/wait.h>
15#include <sys/time.h>
15#include "user.h" 16#include "user.h"
16#include "user_util.h" 17#include "user_util.h"
17#include "kern_util.h" 18#include "kern_util.h"
@@ -258,3 +259,32 @@ char *split_if_spec(char *str, ...)
258 va_end(ap); 259 va_end(ap);
259 return str; 260 return str;
260} 261}
262
263void random_mac(unsigned char *addr)
264{
265 struct timeval tv;
266 long n;
267 unsigned int seed;
268
269 gettimeofday(&tv, NULL);
270
271 /* Assume that 20 bits of microseconds and 12 bits of the pid are
272 * reasonably unpredictable.
273 */
274 seed = tv.tv_usec | (os_getpid() << 20);
275 srandom(seed);
276
277 /* Don't care about endianness here - switching endianness
278 * just rearranges what are hopefully random numbers.
279 *
280 * Assume that RAND_MAX > 65536, so random is called twice and
281 * we use 16 bits of the result.
282 */
283 n = random();
284 addr[2] = (n >> 8) & 255;
285 addr[3] = n % 255;
286
287 n = random();
288 addr[4] = (n >> 8) & 255;
289 addr[5] = n % 255;
290}
diff --git a/arch/um/include/net_user.h b/arch/um/include/net_user.h
index 47ef7cb49a8e..cd19defbfefc 100644
--- a/arch/um/include/net_user.h
+++ b/arch/um/include/net_user.h
@@ -1,4 +1,4 @@
1/* 1/*
2 * Copyright (C) 2002 Jeff Dike (jdike@karaya.com) 2 * Copyright (C) 2002 Jeff Dike (jdike@karaya.com)
3 * Licensed under the GPL 3 * Licensed under the GPL
4 */ 4 */
@@ -26,8 +26,8 @@ struct net_user_info {
26 26
27extern void ether_user_init(void *data, void *dev); 27extern void ether_user_init(void *data, void *dev);
28extern void dev_ip_addr(void *d, unsigned char *bin_buf); 28extern void dev_ip_addr(void *d, unsigned char *bin_buf);
29extern void iter_addresses(void *d, void (*cb)(unsigned char *, 29extern void iter_addresses(void *d, void (*cb)(unsigned char *,
30 unsigned char *, void *), 30 unsigned char *, void *),
31 void *arg); 31 void *arg);
32 32
33extern void *get_output_buffer(int *len_out); 33extern void *get_output_buffer(int *len_out);
@@ -51,15 +51,6 @@ extern char *split_if_spec(char *str, ...);
51 51
52extern int dev_netmask(void *d, void *m); 52extern int dev_netmask(void *d, void *m);
53 53
54#endif 54extern void random_mac(unsigned char *addr);
55 55
56/* 56#endif
57 * Overrides for Emacs so that we follow Linus's tabbing style.
58 * Emacs will notice this stuff at the end of the file and automatically
59 * adjust the settings for this buffer only. This must remain at the end
60 * of the file.
61 * ---------------------------------------------------------------------------
62 * Local variables:
63 * c-file-style: "linux"
64 * End:
65 */