diff options
-rw-r--r-- | networking/udhcp/clientpacket.c | 4 | ||||
-rw-r--r-- | networking/udhcp/domain_codec.c | 73 | ||||
-rw-r--r-- | networking/udhcp/options.c | 2 | ||||
-rw-r--r-- | networking/udhcp/options.h | 8 | ||||
-rw-r--r-- | networking/udhcp/script.c | 37 |
5 files changed, 66 insertions, 58 deletions
diff --git a/networking/udhcp/clientpacket.c b/networking/udhcp/clientpacket.c index 21c1a7bd5..84a6765ee 100644 --- a/networking/udhcp/clientpacket.c +++ b/networking/udhcp/clientpacket.c | |||
@@ -166,7 +166,7 @@ int FAST_FUNC send_select(uint32_t xid, uint32_t server, uint32_t requested) | |||
166 | } | 166 | } |
167 | 167 | ||
168 | 168 | ||
169 | /* Unicasts or broadcasts a DHCP renew message */ | 169 | /* Unicast or broadcast a DHCP renew message */ |
170 | int FAST_FUNC send_renew(uint32_t xid, uint32_t server, uint32_t ciaddr) | 170 | int FAST_FUNC send_renew(uint32_t xid, uint32_t server, uint32_t ciaddr) |
171 | { | 171 | { |
172 | struct dhcp_packet packet; | 172 | struct dhcp_packet packet; |
@@ -186,7 +186,7 @@ int FAST_FUNC send_renew(uint32_t xid, uint32_t server, uint32_t ciaddr) | |||
186 | } | 186 | } |
187 | 187 | ||
188 | 188 | ||
189 | /* Unicasts a DHCP release message */ | 189 | /* Unicast a DHCP release message */ |
190 | int FAST_FUNC send_release(uint32_t server, uint32_t ciaddr) | 190 | int FAST_FUNC send_release(uint32_t server, uint32_t ciaddr) |
191 | { | 191 | { |
192 | struct dhcp_packet packet; | 192 | struct dhcp_packet packet; |
diff --git a/networking/udhcp/domain_codec.c b/networking/udhcp/domain_codec.c index 6f051c4b0..45354e727 100644 --- a/networking/udhcp/domain_codec.c +++ b/networking/udhcp/domain_codec.c | |||
@@ -25,16 +25,9 @@ | |||
25 | */ | 25 | */ |
26 | char* FAST_FUNC dname_dec(const uint8_t *cstr, int clen, const char *pre) | 26 | char* FAST_FUNC dname_dec(const uint8_t *cstr, int clen, const char *pre) |
27 | { | 27 | { |
28 | const uint8_t *c; | 28 | char *ret = ret; /* for compiler */ |
29 | int crtpos, retpos, depth, plen = 0, len = 0; | ||
30 | char *dst = NULL; | 29 | char *dst = NULL; |
31 | 30 | ||
32 | if (!cstr) | ||
33 | return NULL; | ||
34 | |||
35 | if (pre) | ||
36 | plen = strlen(pre); | ||
37 | |||
38 | /* We make two passes over the cstr string. First, we compute | 31 | /* We make two passes over the cstr string. First, we compute |
39 | * how long the resulting string would be. Then we allocate a | 32 | * how long the resulting string would be. Then we allocate a |
40 | * new buffer of the required length, and fill it in with the | 33 | * new buffer of the required length, and fill it in with the |
@@ -42,59 +35,71 @@ char* FAST_FUNC dname_dec(const uint8_t *cstr, int clen, const char *pre) | |||
42 | * having to deal with requiring callers to supply their own | 35 | * having to deal with requiring callers to supply their own |
43 | * buffer, then having to check if it's sufficiently large, etc. | 36 | * buffer, then having to check if it's sufficiently large, etc. |
44 | */ | 37 | */ |
45 | 38 | while (1) { | |
46 | while (!dst) { | 39 | /* note: "return NULL" below are leak-safe since |
47 | 40 | * dst isn't yet allocated */ | |
48 | if (len > 0) { /* second pass? allocate dst buffer and copy pre */ | 41 | const uint8_t *c; |
49 | dst = xmalloc(len + plen); | 42 | unsigned crtpos, retpos, depth, len; |
50 | memcpy(dst, pre, plen); | ||
51 | } | ||
52 | 43 | ||
53 | crtpos = retpos = depth = len = 0; | 44 | crtpos = retpos = depth = len = 0; |
54 | |||
55 | while (crtpos < clen) { | 45 | while (crtpos < clen) { |
56 | c = cstr + crtpos; | 46 | c = cstr + crtpos; |
57 | 47 | ||
58 | if ((*c & NS_CMPRSFLGS) != 0) { /* pointer */ | 48 | if (*c & NS_CMPRSFLGS) { |
59 | if (crtpos + 2 > clen) /* no offset to jump to? abort */ | 49 | /* pointer */ |
50 | if (crtpos + 2 > clen) /* no offset to jump to? abort */ | ||
60 | return NULL; | 51 | return NULL; |
61 | if (retpos == 0) /* toplevel? save return spot */ | 52 | if (retpos == 0) /* toplevel? save return spot */ |
62 | retpos = crtpos + 2; | 53 | retpos = crtpos + 2; |
63 | depth++; | 54 | depth++; |
64 | crtpos = ((*c & 0x3f) << 8) | (*(c + 1) & 0xff); /* jump */ | 55 | crtpos = ((c[0] & 0x3f) << 8) | (c[1] & 0xff); /* jump */ |
65 | } else if (*c) { /* label */ | 56 | } else if (*c) { |
66 | if (crtpos + *c + 1 > clen) /* label too long? abort */ | 57 | /* label */ |
58 | if (crtpos + *c + 1 > clen) /* label too long? abort */ | ||
67 | return NULL; | 59 | return NULL; |
68 | if (dst) | 60 | if (dst) |
69 | memcpy(dst + plen + len, c + 1, *c); | 61 | memcpy(dst + len, c + 1, *c); |
70 | len += *c + 1; | 62 | len += *c + 1; |
71 | crtpos += *c + 1; | 63 | crtpos += *c + 1; |
72 | if (dst) | 64 | if (dst) |
73 | *(dst + plen + len - 1) = '.'; | 65 | dst[len - 1] = '.'; |
74 | } else { /* null: end of current domain name */ | 66 | } else { |
75 | if (retpos == 0) { /* toplevel? keep going */ | 67 | /* null: end of current domain name */ |
68 | if (retpos == 0) { | ||
69 | /* toplevel? keep going */ | ||
76 | crtpos++; | 70 | crtpos++; |
77 | } else { /* return to toplevel saved spot */ | 71 | } else { |
72 | /* return to toplevel saved spot */ | ||
78 | crtpos = retpos; | 73 | crtpos = retpos; |
79 | retpos = depth = 0; | 74 | retpos = depth = 0; |
80 | } | 75 | } |
81 | if (dst) | 76 | if (dst) |
82 | *(dst + plen + len - 1) = ' '; | 77 | dst[len - 1] = ' '; |
83 | } | 78 | } |
84 | 79 | ||
85 | if (depth > NS_MAXDNSRCH || /* too many jumps? abort, it's a loop */ | 80 | if (depth > NS_MAXDNSRCH /* too many jumps? abort, it's a loop */ |
86 | len > NS_MAXDNAME * NS_MAXDNSRCH) /* result too long? abort */ | 81 | || len > NS_MAXDNAME * NS_MAXDNSRCH /* result too long? abort */ |
82 | ) { | ||
87 | return NULL; | 83 | return NULL; |
84 | } | ||
88 | } | 85 | } |
89 | 86 | ||
90 | if (!len) /* expanded string has 0 length? abort */ | 87 | if (!len) /* expanded string has 0 length? abort */ |
91 | return NULL; | 88 | return NULL; |
92 | 89 | ||
93 | if (dst) | 90 | if (!dst) { /* first pass? */ |
94 | *(dst + plen + len - 1) = '\0'; | 91 | /* allocate dst buffer and copy pre */ |
92 | unsigned plen = strlen(pre); | ||
93 | ret = dst = xmalloc(plen + len); | ||
94 | memcpy(dst, pre, plen); | ||
95 | dst += plen; | ||
96 | } else { | ||
97 | dst[len - 1] = '\0'; | ||
98 | break; | ||
99 | } | ||
95 | } | 100 | } |
96 | 101 | ||
97 | return dst; | 102 | return ret; |
98 | } | 103 | } |
99 | 104 | ||
100 | /* Convert a domain name (src) from human-readable "foo.blah.com" format into | 105 | /* Convert a domain name (src) from human-readable "foo.blah.com" format into |
diff --git a/networking/udhcp/options.c b/networking/udhcp/options.c index 0f17feb2a..76d6e3737 100644 --- a/networking/udhcp/options.c +++ b/networking/udhcp/options.c | |||
@@ -115,7 +115,7 @@ const uint8_t dhcp_option_lengths[] ALIGN1 = { | |||
115 | [OPTION_U16] = 2, | 115 | [OPTION_U16] = 2, |
116 | [OPTION_S16] = 2, | 116 | [OPTION_S16] = 2, |
117 | [OPTION_U32] = 4, | 117 | [OPTION_U32] = 4, |
118 | [OPTION_S32] = 4 | 118 | [OPTION_S32] = 4, |
119 | }; | 119 | }; |
120 | 120 | ||
121 | 121 | ||
diff --git a/networking/udhcp/options.h b/networking/udhcp/options.h index 8c80485be..9e624318f 100644 --- a/networking/udhcp/options.h +++ b/networking/udhcp/options.h | |||
@@ -19,11 +19,13 @@ enum { | |||
19 | OPTION_U16, | 19 | OPTION_U16, |
20 | OPTION_S16, | 20 | OPTION_S16, |
21 | OPTION_U32, | 21 | OPTION_U32, |
22 | OPTION_S32 | 22 | OPTION_S32, |
23 | }; | 23 | }; |
24 | 24 | ||
25 | #define OPTION_REQ 0x10 /* have the client request this option */ | 25 | /* Client requests this option by default */ |
26 | #define OPTION_LIST 0x20 /* There can be a list of 1 or more of these */ | 26 | #define OPTION_REQ 0x10 |
27 | /* There can be a list of 1 or more of these */ | ||
28 | #define OPTION_LIST 0x20 | ||
27 | 29 | ||
28 | /*****************************************************************/ | 30 | /*****************************************************************/ |
29 | /* Do not modify below here unless you know what you are doing!! */ | 31 | /* Do not modify below here unless you know what you are doing!! */ |
diff --git a/networking/udhcp/script.c b/networking/udhcp/script.c index 7ebef3553..94dabf4d1 100644 --- a/networking/udhcp/script.c +++ b/networking/udhcp/script.c | |||
@@ -14,7 +14,7 @@ | |||
14 | 14 | ||
15 | 15 | ||
16 | /* get a rough idea of how long an option will be (rounding up...) */ | 16 | /* get a rough idea of how long an option will be (rounding up...) */ |
17 | static const uint8_t max_option_length[] = { | 17 | static const uint8_t len_of_option_as_string[] = { |
18 | [OPTION_IP] = sizeof("255.255.255.255 "), | 18 | [OPTION_IP] = sizeof("255.255.255.255 "), |
19 | [OPTION_IP_PAIR] = sizeof("255.255.255.255 ") * 2, | 19 | [OPTION_IP_PAIR] = sizeof("255.255.255.255 ") * 2, |
20 | [OPTION_STRING] = 1, | 20 | [OPTION_STRING] = 1, |
@@ -30,17 +30,10 @@ static const uint8_t max_option_length[] = { | |||
30 | }; | 30 | }; |
31 | 31 | ||
32 | 32 | ||
33 | static inline int upper_length(int length, int opt_index) | ||
34 | { | ||
35 | return max_option_length[opt_index] * | ||
36 | (length / dhcp_option_lengths[opt_index]); | ||
37 | } | ||
38 | |||
39 | |||
40 | /* note: ip is a pointer to an IP in network order, possibly misaliged */ | 33 | /* note: ip is a pointer to an IP in network order, possibly misaliged */ |
41 | static int sprint_nip(char *dest, const char *pre, const uint8_t *ip) | 34 | static int sprint_nip(char *dest, const char *pre, const uint8_t *ip) |
42 | { | 35 | { |
43 | return sprintf(dest, "%s%d.%d.%d.%d", pre, ip[0], ip[1], ip[2], ip[3]); | 36 | return sprintf(dest, "%s%u.%u.%u.%u", pre, ip[0], ip[1], ip[2], ip[3]); |
44 | } | 37 | } |
45 | 38 | ||
46 | 39 | ||
@@ -57,9 +50,10 @@ static int mton(uint32_t mask) | |||
57 | } | 50 | } |
58 | 51 | ||
59 | 52 | ||
60 | /* Allocate and fill with the text of option 'option'. */ | 53 | /* Create "opt_name=opt_value" string */ |
61 | static char *alloc_fill_opts(uint8_t *option, const struct dhcp_option *type_p, const char *opt_name) | 54 | static char *xmalloc_optname_optval(uint8_t *option, const struct dhcp_option *type_p, const char *opt_name) |
62 | { | 55 | { |
56 | unsigned upper_length; | ||
63 | int len, type, optlen; | 57 | int len, type, optlen; |
64 | uint16_t val_u16; | 58 | uint16_t val_u16; |
65 | int16_t val_s16; | 59 | int16_t val_s16; |
@@ -67,14 +61,16 @@ static char *alloc_fill_opts(uint8_t *option, const struct dhcp_option *type_p, | |||
67 | int32_t val_s32; | 61 | int32_t val_s32; |
68 | char *dest, *ret; | 62 | char *dest, *ret; |
69 | 63 | ||
70 | len = option[OPT_LEN - 2]; | 64 | /* option points to OPT_DATA, need to go back and get OPT_LEN */ |
65 | len = option[OPT_LEN - OPT_DATA]; | ||
71 | type = type_p->flags & TYPE_MASK; | 66 | type = type_p->flags & TYPE_MASK; |
72 | optlen = dhcp_option_lengths[type]; | 67 | optlen = dhcp_option_lengths[type]; |
68 | upper_length = len_of_option_as_string[type] * (len / optlen); | ||
73 | 69 | ||
74 | dest = ret = xmalloc(upper_length(len, type) + strlen(opt_name) + 2); | 70 | dest = ret = xmalloc(upper_length + strlen(opt_name) + 2); |
75 | dest += sprintf(ret, "%s=", opt_name); | 71 | dest += sprintf(ret, "%s=", opt_name); |
76 | 72 | ||
77 | for (;;) { | 73 | while (len >= optlen) { |
78 | switch (type) { | 74 | switch (type) { |
79 | case OPTION_IP_PAIR: | 75 | case OPTION_IP_PAIR: |
80 | dest += sprint_nip(dest, "", option); | 76 | dest += sprint_nip(dest, "", option); |
@@ -114,15 +110,20 @@ static char *alloc_fill_opts(uint8_t *option, const struct dhcp_option *type_p, | |||
114 | case OPTION_STR1035: | 110 | case OPTION_STR1035: |
115 | /* unpack option into dest; use ret for prefix (i.e., "optname=") */ | 111 | /* unpack option into dest; use ret for prefix (i.e., "optname=") */ |
116 | dest = dname_dec(option, len, ret); | 112 | dest = dname_dec(option, len, ret); |
117 | free(ret); | 113 | if (dest) { |
118 | return dest; | 114 | free(ret); |
115 | return dest; | ||
116 | } | ||
117 | /* error. return "optname=" string */ | ||
118 | return ret; | ||
119 | #endif | 119 | #endif |
120 | } | 120 | } |
121 | option += optlen; | 121 | option += optlen; |
122 | len -= optlen; | 122 | len -= optlen; |
123 | if (len <= 0) | 123 | if (len <= 0) |
124 | break; | 124 | break; |
125 | dest += sprintf(dest, " "); | 125 | *dest++ = ' '; |
126 | *dest = '\0'; | ||
126 | } | 127 | } |
127 | return ret; | 128 | return ret; |
128 | } | 129 | } |
@@ -174,7 +175,7 @@ static char **fill_envp(struct dhcp_packet *packet) | |||
174 | temp = get_option(packet, dhcp_options[i].code); | 175 | temp = get_option(packet, dhcp_options[i].code); |
175 | if (!temp) | 176 | if (!temp) |
176 | goto next; | 177 | goto next; |
177 | *curr = alloc_fill_opts(temp, &dhcp_options[i], opt_name); | 178 | *curr = xmalloc_optname_optval(temp, &dhcp_options[i], opt_name); |
178 | putenv(*curr++); | 179 | putenv(*curr++); |
179 | 180 | ||
180 | /* Fill in a subnet bits option for things like /24 */ | 181 | /* Fill in a subnet bits option for things like /24 */ |