/* * Copyright (C) 2017 Freie Universität Berlin * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @{ * * @file * @author Martine Lenders */ #include "net/gnrc/pktbuf.h" gnrc_pktsnip_t *gnrc_pktbuf_remove_snip(gnrc_pktsnip_t *pkt, gnrc_pktsnip_t *snip) { pkt = gnrc_pkt_delete(pkt, snip); snip->next = NULL; gnrc_pktbuf_release(snip); return pkt; } gnrc_pktsnip_t *gnrc_pktbuf_reverse_snips(gnrc_pktsnip_t *pkt) { gnrc_pktsnip_t *reversed = NULL, *ptr = pkt; while (ptr != NULL) { gnrc_pktsnip_t *next; /* try to write-protect snip as its next-pointer is changed below */ pkt = gnrc_pktbuf_start_write(ptr); /* use pkt as temporary variable */ if (pkt == NULL) { gnrc_pktbuf_release(reversed); gnrc_pktbuf_release(ptr); return NULL; } /* switch around pointers */ next = pkt->next; pkt->next = reversed; reversed = pkt; ptr = next; } return reversed; } int gnrc_pktbuf_merge(gnrc_pktsnip_t *pkt) { size_t offset = pkt->size; size_t size = gnrc_pkt_len(pkt); int res = 0; if (pkt->size == size) { return res; } /* Re-allocate data */ res = gnrc_pktbuf_realloc_data(pkt, size); if (res != 0) { return res; } /* Copy data to new buffer */ for (gnrc_pktsnip_t *ptr = pkt->next; ptr != NULL; ptr = ptr->next) { memcpy(((uint8_t *)pkt->data) + offset, ptr->data, ptr->size); offset += ptr->size; } /* Release old pktsnips and data*/ gnrc_pktbuf_release(pkt->next); pkt->next = NULL; return res; } /** @} */