1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-18 12:52:44 +01:00
19713: nanocoap: clean up coap_iterate_option(), make it public r=benpicco a=benpicco



Co-authored-by: Benjamin Valentin <benjamin.valentin@ml-pa.com>
This commit is contained in:
bors[bot] 2023-06-07 22:54:15 +00:00 committed by GitHub
commit ed3b894389
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
3 changed files with 54 additions and 26 deletions

View File

@ -28,6 +28,7 @@ BOARD_INSUFFICIENT_MEMORY := \
stk3200 \ stk3200 \
stm32f030f4-demo \ stm32f030f4-demo \
stm32f0discovery \ stm32f0discovery \
stm32f7508-dk \
stm32g0316-disco \ stm32g0316-disco \
stm32l0538-disco \ stm32l0538-disco \
telosb \ telosb \

View File

@ -656,6 +656,21 @@ static inline void coap_hdr_set_type(coap_hdr_t *hdr, unsigned type)
*/ */
uint8_t *coap_find_option(coap_pkt_t *pkt, unsigned opt_num); uint8_t *coap_find_option(coap_pkt_t *pkt, unsigned opt_num);
/**
* @brief Get pointer to an option field, can be called in a loop
* if there are multiple options with the same number
*
* @param[in] pkt packet to work on
* @param[in] opt_num the option number to search for
* @param[out] opt_pos opaque, must be set to `NULL` on first call
* @param[out] opt_len size of the current option data
*
* @returns pointer to the option data
* NULL if option number was not found
*/
uint8_t *coap_iterate_option(coap_pkt_t *pkt, unsigned opt_num,
uint8_t **opt_pos, int *opt_len);
/** /**
* @brief Get content type from packet * @brief Get content type from packet
* *

View File

@ -281,19 +281,27 @@ int coap_opt_get_uint(coap_pkt_t *pkt, uint16_t opt_num, uint32_t *target)
return -ENOENT; return -ENOENT;
} }
uint8_t *coap_iterate_option(coap_pkt_t *pkt, uint8_t **optpos, uint8_t *coap_iterate_option(coap_pkt_t *pkt, unsigned optnum,
int *opt_len, int first) uint8_t **opt_pos, int *opt_len)
{ {
uint8_t *data_start; uint8_t *data_start;
bool first = false;
if (*opt_pos == NULL) {
*opt_pos = coap_find_option(pkt, optnum);
first = true;
if (*opt_pos == NULL) {
return NULL;
}
}
uint16_t delta = 0; uint16_t delta = 0;
data_start = _parse_option(pkt, *optpos, &delta, opt_len); data_start = _parse_option(pkt, *opt_pos, &delta, opt_len);
if (data_start && (first || !delta)) { if (data_start && (first || !delta)) {
*optpos = data_start + *opt_len; *opt_pos = data_start + *opt_len;
return data_start; return data_start;
} }
else { else {
*optpos = NULL;
return NULL; return NULL;
} }
} }
@ -359,31 +367,35 @@ ssize_t coap_opt_get_string(coap_pkt_t *pkt, uint16_t optnum,
{ {
assert(pkt && target && (max_len > 1)); assert(pkt && target && (max_len > 1));
uint8_t *opt_pos = coap_find_option(pkt, optnum); uint8_t *opt_pos = NULL;
if (!opt_pos) { int opt_len;
unsigned left = max_len;
while (1) {
uint8_t *part_start = coap_iterate_option(pkt, optnum, &opt_pos, &opt_len);
if (part_start == NULL) {
/* if option was not found still return separator */
if (opt_pos == NULL) {
*target++ = (uint8_t)separator;
--left;
}
break;
}
/* separator and terminating \0 have to fit */
if (left < (unsigned)(opt_len + 2)) {
return -ENOSPC;
}
*target++ = (uint8_t)separator; *target++ = (uint8_t)separator;
*target = '\0'; memcpy(target, part_start, opt_len);
return 2; target += opt_len;
left -= opt_len + 1;
} }
unsigned left = max_len - 1;
uint8_t *part_start = NULL;
do {
int opt_len;
part_start = coap_iterate_option(pkt, &opt_pos, &opt_len,
(part_start == NULL));
if (part_start) {
if (left < (unsigned)(opt_len + 1)) {
return -ENOSPC;
}
*target++ = (uint8_t)separator;
memcpy(target, part_start, opt_len);
target += opt_len;
left -= (opt_len + 1);
}
} while (opt_pos);
*target = '\0'; *target = '\0';
left--;
return (int)(max_len - left); return (int)(max_len - left);
} }