aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--networking/udhcp/clientpacket.c4
-rw-r--r--networking/udhcp/domain_codec.c73
-rw-r--r--networking/udhcp/options.c2
-rw-r--r--networking/udhcp/options.h8
-rw-r--r--networking/udhcp/script.c37
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 */
170int FAST_FUNC send_renew(uint32_t xid, uint32_t server, uint32_t ciaddr) 170int 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 */
190int FAST_FUNC send_release(uint32_t server, uint32_t ciaddr) 190int 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 */
26char* FAST_FUNC dname_dec(const uint8_t *cstr, int clen, const char *pre) 26char* 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...) */
17static const uint8_t max_option_length[] = { 17static 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
33static 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 */
41static int sprint_nip(char *dest, const char *pre, const uint8_t *ip) 34static 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 */
61static char *alloc_fill_opts(uint8_t *option, const struct dhcp_option *type_p, const char *opt_name) 54static 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 */