899 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			899 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2016 MariaDB Corporation Ab
 | |
|  *
 | |
|  * Use of this software is governed by the Business Source License included
 | |
|  * in the LICENSE.TXT file and at www.mariadb.com/bsl11.
 | |
|  *
 | |
|  * Change Date: 2024-08-24
 | |
|  *
 | |
|  * On the date above, in accordance with the Business Source License, use
 | |
|  * of this software will be governed by version 2 or later of the General
 | |
|  * Public License.
 | |
|  */
 | |
| 
 | |
| #include <maxscale/buffer.hh>
 | |
| 
 | |
| #include <errno.h>
 | |
| #include <stdlib.h>
 | |
| #include <sstream>
 | |
| 
 | |
| #include <maxbase/assert.h>
 | |
| #include <maxbase/alloc.h>
 | |
| #include <maxscale/hint.h>
 | |
| #include <maxscale/utils.h>
 | |
| #include <maxscale/routingworker.hh>
 | |
| 
 | |
| using mxs::RoutingWorker;
 | |
| 
 | |
| static void             gwbuf_free_one(GWBUF* buf);
 | |
| static buffer_object_t* gwbuf_remove_buffer_object(GWBUF* buf,
 | |
|                                                    buffer_object_t* bufobj);
 | |
| 
 | |
| #if defined (SS_DEBUG)
 | |
| inline void invalidate_tail_pointers(GWBUF* head)
 | |
| {
 | |
|     if (head && head->next)
 | |
|     {
 | |
|         GWBUF* link = head->next;
 | |
|         while (link != head->tail)
 | |
|         {
 | |
|             link->tail = reinterpret_cast<GWBUF*>(0xdeadbeef);
 | |
|             link = link->next;
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| inline void ensure_at_head(const GWBUF* buf)
 | |
| {
 | |
|     mxb_assert(buf->tail != reinterpret_cast<GWBUF*>(0xdeadbeef));
 | |
| }
 | |
| 
 | |
| inline void ensure_owned(const GWBUF* buf)
 | |
| {
 | |
|     mxb_assert(buf->owner == RoutingWorker::get_current_id());
 | |
| }
 | |
| 
 | |
| inline bool validate_buffer(const GWBUF* buf)
 | |
| {
 | |
|     mxb_assert(buf);
 | |
|     ensure_at_head(buf);
 | |
|     ensure_owned(buf);
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| #else
 | |
| inline void invalidate_tail_pointers(GWBUF* head)
 | |
| {
 | |
| }
 | |
| 
 | |
| inline void ensure_at_head(const GWBUF* head)
 | |
| {
 | |
| }
 | |
| 
 | |
| inline void ensure_owned(const GWBUF* head)
 | |
| {
 | |
| }
 | |
| 
 | |
| inline bool validate_buffer(const GWBUF* head)
 | |
| {
 | |
|     return true;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| /**
 | |
|  * Allocate a new gateway buffer structure of size bytes.
 | |
|  *
 | |
|  * For now we allocate memory directly from malloc for buffer the management
 | |
|  * structure and the actual data buffer itself. We may swap at a future date
 | |
|  * to a more efficient mechanism.
 | |
|  *
 | |
|  * @param       size The size in bytes of the data area required
 | |
|  * @return      Pointer to the buffer structure or NULL if memory could not
 | |
|  *              be allocated.
 | |
|  */
 | |
| GWBUF* gwbuf_alloc(unsigned int size)
 | |
| {
 | |
|     size_t sbuf_size = sizeof(SHARED_BUF) + (size ? size - 1 : 0);
 | |
|     GWBUF* rval = (GWBUF*)MXS_MALLOC(sizeof(GWBUF));
 | |
|     SHARED_BUF* sbuf = (SHARED_BUF*)MXS_MALLOC(sbuf_size);
 | |
| 
 | |
|     if (rval == NULL || sbuf == NULL)
 | |
|     {
 | |
|         MXS_FREE(rval);
 | |
|         MXS_FREE(sbuf);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     sbuf->refcount = 1;
 | |
|     sbuf->info = GWBUF_INFO_NONE;
 | |
|     sbuf->bufobj = NULL;
 | |
| 
 | |
| #ifdef SS_DEBUG
 | |
|     rval->owner = RoutingWorker::get_current_id();
 | |
| #endif
 | |
|     rval->start = &sbuf->data;
 | |
|     rval->end = (void*)((char*)rval->start + size);
 | |
|     rval->sbuf = sbuf;
 | |
|     rval->next = NULL;
 | |
|     rval->tail = rval;
 | |
|     rval->hint = NULL;
 | |
|     rval->properties = NULL;
 | |
|     rval->gwbuf_type = GWBUF_TYPE_UNDEFINED;
 | |
|     rval->server = NULL;
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Allocate a new gateway buffer structure of size bytes and load with data.
 | |
|  *
 | |
|  * @param       size    The size in bytes of the data area required
 | |
|  * @param       data    Pointer to the data (size bytes) to be loaded
 | |
|  * @return      Pointer to the buffer structure or NULL if memory could not
 | |
|  *              be allocated.
 | |
|  */
 | |
| GWBUF* gwbuf_alloc_and_load(unsigned int size, const void* data)
 | |
| {
 | |
|     GWBUF* rval = gwbuf_alloc(size);
 | |
| 
 | |
|     if (rval)
 | |
|     {
 | |
|         memcpy(GWBUF_DATA(rval), data, size);
 | |
|     }
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Free a list of gateway buffers
 | |
|  *
 | |
|  * @param buf The head of the list of buffers to free
 | |
|  */
 | |
| void gwbuf_free(GWBUF* buf)
 | |
| {
 | |
|     mxb_assert(!buf || validate_buffer(buf));
 | |
| 
 | |
|     while (buf)
 | |
|     {
 | |
|         GWBUF* nextbuf = buf->next;
 | |
|         gwbuf_free_one(buf);
 | |
|         buf = nextbuf;
 | |
|     }
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Free a single gateway buffer
 | |
|  *
 | |
|  * @param buf The buffer to free
 | |
|  */
 | |
| static void gwbuf_free_one(GWBUF* buf)
 | |
| {
 | |
|     ensure_owned(buf);
 | |
| 
 | |
|     --buf->sbuf->refcount;
 | |
| 
 | |
|     if (buf->sbuf->refcount == 0)
 | |
|     {
 | |
|         buffer_object_t* bo = buf->sbuf->bufobj;
 | |
| 
 | |
|         while (bo != NULL)
 | |
|         {
 | |
|             bo = gwbuf_remove_buffer_object(buf, bo);
 | |
|         }
 | |
| 
 | |
|         MXS_FREE(buf->sbuf);
 | |
|     }
 | |
| 
 | |
|     while (buf->properties)
 | |
|     {
 | |
|         BUF_PROPERTY* prop = buf->properties;
 | |
|         buf->properties = prop->next;
 | |
|         MXS_FREE(prop->name);
 | |
|         MXS_FREE(prop->value);
 | |
|         MXS_FREE(prop);
 | |
|     }
 | |
|     /** Release the hint */
 | |
|     while (buf->hint)
 | |
|     {
 | |
|         HINT* h = buf->hint;
 | |
|         buf->hint = buf->hint->next;
 | |
|         hint_free(h);
 | |
|     }
 | |
| 
 | |
|     MXS_FREE(buf);
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Increment the usage count of a gateway buffer. This gets a new
 | |
|  * GWBUF structure that shares the actual data with the existing
 | |
|  * GWBUF structure but allows for the data copy to be avoided and
 | |
|  * also for each GWBUF to point to different portions of the same
 | |
|  * SHARED_BUF.
 | |
|  *
 | |
|  * @param buf The buffer to use
 | |
|  * @return A new GWBUF structure
 | |
|  */
 | |
| static GWBUF* gwbuf_clone_one(GWBUF* buf)
 | |
| {
 | |
|     GWBUF* rval = (GWBUF*)MXS_CALLOC(1, sizeof(GWBUF));
 | |
| 
 | |
|     if (rval == NULL)
 | |
|     {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     mxb_assert(buf->owner == RoutingWorker::get_current_id());
 | |
|     ++buf->sbuf->refcount;
 | |
| #ifdef SS_DEBUG
 | |
|     rval->owner = RoutingWorker::get_current_id();
 | |
| #endif
 | |
|     rval->server = buf->server;
 | |
|     rval->sbuf = buf->sbuf;
 | |
|     rval->start = buf->start;
 | |
|     rval->end = buf->end;
 | |
|     rval->gwbuf_type = buf->gwbuf_type;
 | |
|     rval->tail = rval;
 | |
|     rval->hint = hint_dup(buf->hint);
 | |
|     rval->next = NULL;
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| GWBUF* gwbuf_clone(GWBUF* buf)
 | |
| {
 | |
|     validate_buffer(buf);
 | |
| 
 | |
|     GWBUF* rval = gwbuf_clone_one(buf);
 | |
| 
 | |
|     if (rval)
 | |
|     {
 | |
|         GWBUF* clonebuf = rval;
 | |
| 
 | |
|         while (clonebuf && buf->next)
 | |
|         {
 | |
|             buf = buf->next;
 | |
|             clonebuf->next = gwbuf_clone_one(buf);
 | |
|             clonebuf = clonebuf->next;
 | |
|         }
 | |
| 
 | |
|         if (!clonebuf && buf->next)
 | |
|         {
 | |
|             // A gwbuf_clone failed, we need to free everything cloned sofar.
 | |
|             gwbuf_free(rval);
 | |
|             rval = NULL;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             rval->tail = clonebuf;
 | |
|         }
 | |
| 
 | |
|         invalidate_tail_pointers(rval);
 | |
|     }
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| static GWBUF* gwbuf_deep_clone_portion(const GWBUF* buf, size_t length)
 | |
| {
 | |
|     ensure_owned(buf);
 | |
|     GWBUF* rval = NULL;
 | |
| 
 | |
|     if (buf)
 | |
|     {
 | |
|         rval = gwbuf_alloc(length);
 | |
| 
 | |
|         if (rval && gwbuf_copy_data(buf, 0, length, GWBUF_DATA(rval)) == length)
 | |
|         {
 | |
|             // The copying of the type is done to retain the type characteristic of the buffer without
 | |
|             // having a link the orginal data or parsing info.
 | |
|             rval->gwbuf_type = buf->gwbuf_type;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             gwbuf_free(rval);
 | |
|             rval = NULL;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| GWBUF* gwbuf_deep_clone(const GWBUF* buf)
 | |
| {
 | |
|     validate_buffer(buf);
 | |
|     return gwbuf_deep_clone_portion(buf, gwbuf_length(buf));
 | |
| }
 | |
| 
 | |
| static GWBUF* gwbuf_clone_portion(GWBUF* buf,
 | |
|                                   size_t start_offset,
 | |
|                                   size_t length)
 | |
| {
 | |
|     ensure_owned(buf);
 | |
|     mxb_assert(start_offset + length <= GWBUF_LENGTH(buf));
 | |
| 
 | |
|     GWBUF* clonebuf = (GWBUF*)MXS_MALLOC(sizeof(GWBUF));
 | |
| 
 | |
|     if (clonebuf == NULL)
 | |
|     {
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     ++buf->sbuf->refcount;
 | |
| #ifdef SS_DEBUG
 | |
|     clonebuf->owner = RoutingWorker::get_current_id();
 | |
| #endif
 | |
|     clonebuf->server = buf->server;
 | |
|     clonebuf->sbuf = buf->sbuf;
 | |
|     clonebuf->gwbuf_type = buf->gwbuf_type;     /*< clone info bits too */
 | |
|     clonebuf->start = (void*)((char*)buf->start + start_offset);
 | |
|     clonebuf->end = (void*)((char*)clonebuf->start + length);
 | |
|     clonebuf->gwbuf_type = buf->gwbuf_type;     /*< clone the type for now */
 | |
|     clonebuf->properties = NULL;
 | |
|     clonebuf->hint = NULL;
 | |
|     clonebuf->next = NULL;
 | |
|     clonebuf->tail = clonebuf;
 | |
| 
 | |
|     return clonebuf;
 | |
| }
 | |
| 
 | |
| GWBUF* gwbuf_split(GWBUF** buf, size_t length)
 | |
| {
 | |
|     validate_buffer(*buf);
 | |
|     GWBUF* head = NULL;
 | |
| 
 | |
|     if (length > 0 && buf && *buf)
 | |
|     {
 | |
|         GWBUF* buffer = *buf;
 | |
|         GWBUF* orig_tail = buffer->tail;
 | |
|         head = buffer;
 | |
|         ensure_owned(buffer);
 | |
| 
 | |
|         /** Handle complete buffers */
 | |
|         while (buffer && length && length >= GWBUF_LENGTH(buffer))
 | |
|         {
 | |
|             length -= GWBUF_LENGTH(buffer);
 | |
|             head->tail = buffer;
 | |
|             buffer = buffer->next;
 | |
|         }
 | |
| 
 | |
|         /** Some data is left in the original buffer */
 | |
|         if (buffer)
 | |
|         {
 | |
|             /** We're splitting a chain of buffers */
 | |
|             if (head->tail != orig_tail)
 | |
|             {
 | |
|                 /** Make sure the original buffer's tail points to the right place */
 | |
|                 buffer->tail = orig_tail;
 | |
| 
 | |
|                 /** Remove the pointer to the original buffer */
 | |
|                 head->tail->next = NULL;
 | |
|             }
 | |
| 
 | |
|             if (length > 0)
 | |
|             {
 | |
|                 mxb_assert(GWBUF_LENGTH(buffer) > length);
 | |
|                 GWBUF* partial = gwbuf_deep_clone_portion(buffer, length);
 | |
| 
 | |
|                 /** If the head points to the original head of the buffer chain
 | |
|                  * and we are splitting a contiguous buffer, we only need to return
 | |
|                  * the partial clone of the first buffer. If we are splitting multiple
 | |
|                  * buffers, we need to append them to the full buffers. */
 | |
|                 head = head == buffer ? partial : gwbuf_append(head, partial);
 | |
| 
 | |
|                 buffer = gwbuf_consume(buffer, length);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         *buf = buffer;
 | |
| 
 | |
|         invalidate_tail_pointers(*buf);
 | |
|         invalidate_tail_pointers(head);
 | |
|     }
 | |
| 
 | |
|     return head;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * Get a byte from a GWBUF at a particular offset. Intended to be use like:
 | |
|  *
 | |
|  *     GWBUF *buf = ...;
 | |
|  *     size_t offset = 0;
 | |
|  *     uint8_t c;
 | |
|  *
 | |
|  *     while (gwbuf_get_byte(&buf, &offset, &c))
 | |
|  *     {
 | |
|  *         printf("%c", c);
 | |
|  *     }
 | |
|  *
 | |
|  * @param buf     Pointer to pointer to GWBUF. The GWBUF pointed to may be adjusted
 | |
|  *                as a result of the call.
 | |
|  * @param offset  Pointer to variable containing the offset. Value of variable will
 | |
|  *                incremented as a result of the call.
 | |
|  * @param b       Pointer to variable that upon successful return will contain the
 | |
|  *                next byte.
 | |
|  *
 | |
|  * @return True, if offset refers to a byte in the GWBUF.
 | |
|  */
 | |
| static inline bool gwbuf_get_byte(const GWBUF** buf, size_t* offset, uint8_t* b)
 | |
| {
 | |
|     bool rv = false;
 | |
| 
 | |
|     // Ignore NULL buffer and walk past empty or too short buffers.
 | |
|     while (*buf && (GWBUF_LENGTH(*buf) <= *offset))
 | |
|     {
 | |
|         mxb_assert((*buf)->owner == RoutingWorker::get_current_id());
 | |
|         *offset -= GWBUF_LENGTH(*buf);
 | |
|         *buf = (*buf)->next;
 | |
|     }
 | |
| 
 | |
|     mxb_assert(!*buf || (GWBUF_LENGTH(*buf) > *offset));
 | |
| 
 | |
|     if (*buf)
 | |
|     {
 | |
|         mxb_assert((*buf)->owner == RoutingWorker::get_current_id());
 | |
|         *b = *(GWBUF_DATA(*buf) + *offset);
 | |
|         *offset += 1;
 | |
| 
 | |
|         rv = true;
 | |
|     }
 | |
| 
 | |
|     return rv;
 | |
| }
 | |
| 
 | |
| int gwbuf_compare(const GWBUF* lhs, const GWBUF* rhs)
 | |
| {
 | |
|     validate_buffer(lhs);
 | |
|     validate_buffer(rhs);
 | |
| 
 | |
|     int rv;
 | |
| 
 | |
|     size_t llen = gwbuf_length(lhs);
 | |
|     size_t rlen = gwbuf_length(rhs);
 | |
| 
 | |
|     if (llen < rlen)
 | |
|     {
 | |
|         rv = -1;
 | |
|     }
 | |
|     else if (rlen < llen)
 | |
|     {
 | |
|         rv = 1;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         mxb_assert(llen == rlen);
 | |
| 
 | |
|         rv = 0;
 | |
|         size_t i = 0;
 | |
|         size_t loffset = 0;
 | |
|         size_t roffset = 0;
 | |
| 
 | |
|         while ((rv == 0) && (i < llen))
 | |
|         {
 | |
|             uint8_t lc = 0;
 | |
|             uint8_t rc = 0;
 | |
| 
 | |
|             MXB_AT_DEBUG(bool rv1 = ) gwbuf_get_byte(&lhs, &loffset, &lc);
 | |
|             MXB_AT_DEBUG(bool rv2 = ) gwbuf_get_byte(&rhs, &roffset, &rc);
 | |
| 
 | |
|             mxb_assert(rv1 && rv2);
 | |
| 
 | |
|             rv = (int)lc - (int)rc;
 | |
| 
 | |
|             ++i;
 | |
|         }
 | |
| 
 | |
|         if (rv < 0)
 | |
|         {
 | |
|             rv = -1;
 | |
|         }
 | |
|         else if (rv > 0)
 | |
|         {
 | |
|             rv = 1;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return rv;
 | |
| }
 | |
| 
 | |
| GWBUF* gwbuf_append(GWBUF* head, GWBUF* tail)
 | |
| {
 | |
|     mxb_assert(!head || validate_buffer(head));
 | |
|     mxb_assert(validate_buffer(tail));
 | |
| 
 | |
|     if (!head)
 | |
|     {
 | |
|         return tail;
 | |
|     }
 | |
| 
 | |
|     head->tail->next = tail;
 | |
|     head->tail = tail->tail;
 | |
| 
 | |
|     invalidate_tail_pointers(head);
 | |
| 
 | |
|     return head;
 | |
| }
 | |
| 
 | |
| GWBUF* gwbuf_consume(GWBUF* head, unsigned int length)
 | |
| {
 | |
|     validate_buffer(head);
 | |
| 
 | |
|     while (head && length > 0)
 | |
|     {
 | |
|         mxb_assert(head->owner == RoutingWorker::get_current_id());
 | |
|         unsigned int buflen = GWBUF_LENGTH(head);
 | |
| 
 | |
|         GWBUF_CONSUME(head, length);
 | |
|         length = buflen < length ? length - buflen : 0;
 | |
| 
 | |
|         if (GWBUF_EMPTY(head))
 | |
|         {
 | |
|             if (head->next)
 | |
|             {
 | |
|                 head->next->tail = head->tail;
 | |
|             }
 | |
|             GWBUF* tmp = head;
 | |
|             head = head->next;
 | |
|             gwbuf_free_one(tmp);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     invalidate_tail_pointers(head);
 | |
| 
 | |
|     mxb_assert(head == NULL || (head->end >= head->start));
 | |
|     return head;
 | |
| }
 | |
| 
 | |
| unsigned int gwbuf_length(const GWBUF* head)
 | |
| {
 | |
|     validate_buffer(head);
 | |
| 
 | |
|     int rval = 0;
 | |
| 
 | |
|     while (head)
 | |
|     {
 | |
|         ensure_owned(head);
 | |
|         rval += GWBUF_LENGTH(head);
 | |
|         head = head->next;
 | |
|     }
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| int gwbuf_count(const GWBUF* head)
 | |
| {
 | |
|     validate_buffer(head);
 | |
| 
 | |
|     int result = 0;
 | |
| 
 | |
|     while (head)
 | |
|     {
 | |
|         ensure_owned(head);
 | |
|         result++;
 | |
|         head = head->next;
 | |
|     }
 | |
| 
 | |
|     return result;
 | |
| }
 | |
| 
 | |
| GWBUF* gwbuf_rtrim(GWBUF* head, unsigned int n_bytes)
 | |
| {
 | |
|     validate_buffer(head);
 | |
| 
 | |
|     GWBUF* rval = head;
 | |
|     GWBUF_RTRIM(head, n_bytes);
 | |
| 
 | |
|     if (GWBUF_EMPTY(head))
 | |
|     {
 | |
|         rval = head->next;
 | |
|         gwbuf_free_one(head);
 | |
|     }
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| void gwbuf_set_type(GWBUF* buf, uint32_t type)
 | |
| {
 | |
|     validate_buffer(buf);
 | |
|     /** Set type consistenly to all buffers on the list */
 | |
|     while (buf != NULL)
 | |
|     {
 | |
|         mxb_assert(buf->owner == RoutingWorker::get_current_id());
 | |
|         buf->gwbuf_type |= type;
 | |
|         buf = buf->next;
 | |
|     }
 | |
| }
 | |
| 
 | |
| void gwbuf_add_buffer_object(GWBUF* buf,
 | |
|                              bufobj_id_t id,
 | |
|                              void* data,
 | |
|                              void (* donefun_fp)(void*))
 | |
| {
 | |
|     validate_buffer(buf);
 | |
| 
 | |
|     buffer_object_t* newb = (buffer_object_t*)MXS_MALLOC(sizeof(buffer_object_t));
 | |
|     MXS_ABORT_IF_NULL(newb);
 | |
| 
 | |
|     newb->bo_id = id;
 | |
|     newb->bo_data = data;
 | |
|     newb->bo_donefun_fp = donefun_fp;
 | |
|     newb->bo_next = NULL;
 | |
| 
 | |
|     buffer_object_t** p_b = &buf->sbuf->bufobj;
 | |
|     /** Search the end of the list and add there */
 | |
|     while (*p_b != NULL)
 | |
|     {
 | |
|         p_b = &(*p_b)->bo_next;
 | |
|     }
 | |
|     *p_b = newb;
 | |
|     /** Set flag */
 | |
|     buf->sbuf->info |= GWBUF_INFO_PARSED;
 | |
| }
 | |
| 
 | |
| void* gwbuf_get_buffer_object_data(GWBUF* buf, bufobj_id_t id)
 | |
| {
 | |
|     validate_buffer(buf);
 | |
| 
 | |
|     buffer_object_t* bo = buf->sbuf->bufobj;
 | |
| 
 | |
|     while (bo != NULL && bo->bo_id != id)
 | |
|     {
 | |
|         bo = bo->bo_next;
 | |
|     }
 | |
| 
 | |
|     return bo ? bo->bo_data : NULL;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @return pointer to next buffer object or NULL
 | |
|  */
 | |
| static buffer_object_t* gwbuf_remove_buffer_object(GWBUF* buf, buffer_object_t* bufobj)
 | |
| {
 | |
|     ensure_owned(buf);
 | |
|     buffer_object_t* next = bufobj->bo_next;
 | |
|     /** Call corresponding clean-up function to clean buffer object's data */
 | |
|     bufobj->bo_donefun_fp(bufobj->bo_data);
 | |
|     MXS_FREE(bufobj);
 | |
|     return next;
 | |
| }
 | |
| 
 | |
| bool gwbuf_add_property(GWBUF* buf, const char* name, const char* value)
 | |
| {
 | |
|     validate_buffer(buf);
 | |
| 
 | |
|     char* my_name = MXS_STRDUP(name);
 | |
|     char* my_value = MXS_STRDUP(value);
 | |
|     BUF_PROPERTY* prop = (BUF_PROPERTY*)MXS_MALLOC(sizeof(BUF_PROPERTY));
 | |
| 
 | |
|     if (!my_name || !my_value || !prop)
 | |
|     {
 | |
|         MXS_FREE(my_name);
 | |
|         MXS_FREE(my_value);
 | |
|         MXS_FREE(prop);
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     prop->name = my_name;
 | |
|     prop->value = my_value;
 | |
|     prop->next = buf->properties;
 | |
|     buf->properties = prop;
 | |
| 
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| char* gwbuf_get_property(GWBUF* buf, const char* name)
 | |
| {
 | |
|     validate_buffer(buf);
 | |
| 
 | |
|     BUF_PROPERTY* prop = buf->properties;
 | |
| 
 | |
|     while (prop && strcmp(prop->name, name) != 0)
 | |
|     {
 | |
|         prop = prop->next;
 | |
|     }
 | |
| 
 | |
|     return prop ? prop->value : NULL;
 | |
| }
 | |
| 
 | |
| GWBUF* gwbuf_make_contiguous(GWBUF* orig)
 | |
| {
 | |
|     validate_buffer(orig);
 | |
| 
 | |
|     if (orig->next == NULL)
 | |
|     {
 | |
|         // Already contiguous
 | |
|         return orig;
 | |
|     }
 | |
| 
 | |
|     GWBUF* newbuf = gwbuf_alloc(gwbuf_length(orig));
 | |
|     MXS_ABORT_IF_NULL(newbuf);
 | |
| 
 | |
|     newbuf->gwbuf_type = orig->gwbuf_type;
 | |
|     newbuf->hint = hint_dup(orig->hint);
 | |
|     uint8_t* ptr = GWBUF_DATA(newbuf);
 | |
| 
 | |
|     while (orig)
 | |
|     {
 | |
|         int len = GWBUF_LENGTH(orig);
 | |
|         memcpy(ptr, GWBUF_DATA(orig), len);
 | |
|         ptr += len;
 | |
|         orig = gwbuf_consume(orig, len);
 | |
|     }
 | |
| 
 | |
|     return newbuf;
 | |
| }
 | |
| 
 | |
| size_t gwbuf_copy_data(const GWBUF* buffer, size_t offset, size_t bytes, uint8_t* dest)
 | |
| {
 | |
|     uint32_t buflen;
 | |
| 
 | |
|     /** Skip unrelated buffers */
 | |
|     while (buffer && offset && offset >= (buflen = GWBUF_LENGTH(buffer)))
 | |
|     {
 | |
|         mxb_assert(buffer->owner == RoutingWorker::get_current_id());
 | |
|         offset -= buflen;
 | |
|         buffer = buffer->next;
 | |
|     }
 | |
| 
 | |
|     size_t bytes_read = 0;
 | |
| 
 | |
|     if (buffer)
 | |
|     {
 | |
|         mxb_assert(buffer->owner == RoutingWorker::get_current_id());
 | |
|         uint8_t* ptr = (uint8_t*) GWBUF_DATA(buffer) + offset;
 | |
|         uint32_t bytes_left = GWBUF_LENGTH(buffer) - offset;
 | |
| 
 | |
|         /** Data is in one buffer */
 | |
|         if (bytes_left >= bytes)
 | |
|         {
 | |
|             memcpy(dest, ptr, bytes);
 | |
|             bytes_read = bytes;
 | |
|         }
 | |
|         else
 | |
|         {
 | |
|             /** Data is spread across multiple buffers */
 | |
|             do
 | |
|             {
 | |
|                 memcpy(dest, ptr, bytes_left);
 | |
|                 bytes -= bytes_left;
 | |
|                 dest += bytes_left;
 | |
|                 bytes_read += bytes_left;
 | |
|                 buffer = buffer->next;
 | |
| 
 | |
|                 if (buffer)
 | |
|                 {
 | |
|                     bytes_left = MXS_MIN(GWBUF_LENGTH(buffer), bytes);
 | |
|                     ptr = (uint8_t*) GWBUF_DATA(buffer);
 | |
|                 }
 | |
|             }
 | |
|             while (bytes > 0 && buffer);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     return bytes_read;
 | |
| }
 | |
| 
 | |
| uint8_t* gwbuf_byte_pointer(GWBUF* buffer, size_t offset)
 | |
| {
 | |
|     validate_buffer(buffer);
 | |
|     uint8_t* rval = NULL;
 | |
|     // Ignore NULL buffer and walk past empty or too short buffers.
 | |
|     while (buffer && (GWBUF_LENGTH(buffer) <= offset))
 | |
|     {
 | |
|         mxb_assert(buffer->owner == RoutingWorker::get_current_id());
 | |
|         offset -= GWBUF_LENGTH(buffer);
 | |
|         buffer = buffer->next;
 | |
|     }
 | |
| 
 | |
|     if (buffer != NULL)
 | |
|     {
 | |
|         mxb_assert(buffer->owner == RoutingWorker::get_current_id());
 | |
|         rval = (GWBUF_DATA(buffer) + offset);
 | |
|     }
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| static std::string dump_one_buffer(GWBUF* buffer)
 | |
| {
 | |
|     ensure_owned(buffer);
 | |
|     std::string rval;
 | |
|     int len = GWBUF_LENGTH(buffer);
 | |
|     uint8_t* data = GWBUF_DATA(buffer);
 | |
| 
 | |
|     while (len > 0)
 | |
|     {
 | |
|         // Process the buffer in 40 byte chunks
 | |
|         int n = MXS_MIN(40, len);
 | |
|         char output[n * 2 + 1];
 | |
|         gw_bin2hex(output, data, n);
 | |
|         char* ptr = output;
 | |
| 
 | |
|         while (ptr < output + n * 2)
 | |
|         {
 | |
|             rval.append(ptr, 2);
 | |
|             rval += " ";
 | |
|             ptr += 2;
 | |
|         }
 | |
|         len -= n;
 | |
|         data += n;
 | |
|         rval += "\n";
 | |
|     }
 | |
| 
 | |
|     return rval;
 | |
| }
 | |
| 
 | |
| void gwbuf_hexdump(GWBUF* buffer, int log_level)
 | |
| {
 | |
|     validate_buffer(buffer);
 | |
|     mxb_assert(buffer->owner == RoutingWorker::get_current_id());
 | |
|     std::stringstream ss;
 | |
| 
 | |
|     ss << "Buffer " << buffer << ":\n";
 | |
| 
 | |
|     for (GWBUF* b = buffer; b; b = b->next)
 | |
|     {
 | |
|         ss << dump_one_buffer(b);
 | |
|     }
 | |
| 
 | |
|     int n = ss.str().length();
 | |
| 
 | |
|     if (n > 1024)
 | |
|     {
 | |
|         n = 1024;
 | |
|     }
 | |
| 
 | |
|     MXS_LOG_MESSAGE(log_level, "%.*s", n, ss.str().c_str());
 | |
| }
 | |
| 
 | |
| void gwbuf_hexdump_pretty(GWBUF* buffer, int log_level)
 | |
| {
 | |
|     mxs::Buffer buf(buffer);
 | |
|     buf.hexdump_pretty(log_level);
 | |
|     buf.release();
 | |
| }
 | |
| 
 | |
| void mxs::Buffer::hexdump(int log_level) const
 | |
| {
 | |
|     return gwbuf_hexdump(m_pBuffer, log_level);
 | |
| }
 | |
| 
 | |
| void mxs::Buffer::hexdump_pretty(int log_level) const
 | |
| {
 | |
|     constexpr const char as_hex[] = "0123456789abcdefghijklmnopqrstuvwxyz";
 | |
|     std::string result = "\n";
 | |
|     std::string hexed;
 | |
|     std::string readable;
 | |
|     auto it = begin();
 | |
| 
 | |
|     while (it != end())
 | |
|     {
 | |
|         for (int i = 0; i < 16 && it != end(); i++)
 | |
|         {
 | |
|             uint8_t c = *it;
 | |
|             hexed += as_hex[c >> 4];
 | |
|             hexed += as_hex[c & 0x0f];
 | |
|             hexed += ' ';
 | |
|             readable += isprint(c) && (!isspace(c) || c == ' ') ? (char)c : '.';
 | |
|             ++it;
 | |
|         }
 | |
| 
 | |
|         if (readable.length() < 16)
 | |
|         {
 | |
|             hexed.append(48 - hexed.length(), ' ');
 | |
|             readable.append(16 - readable.length(), ' ');
 | |
|         }
 | |
| 
 | |
|         mxb_assert(hexed.length() == readable.length() * 3);
 | |
|         result += hexed.substr(0, 24);
 | |
|         result += "  ";
 | |
|         result += hexed.substr(24);
 | |
|         result += "  ";
 | |
|         result += readable;
 | |
|         result += '\n';
 | |
| 
 | |
|         hexed.clear();
 | |
|         readable.clear();
 | |
|     }
 | |
| 
 | |
|     MXS_LOG_MESSAGE(log_level, "%s", result.c_str());
 | |
| }
 | 
