xref: /openbmc/u-boot/lib/circbuf.c (revision e8f80a5a)
1*83d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+
278acc472SPeter Tyser /*
378acc472SPeter Tyser  * (C) Copyright 2003
478acc472SPeter Tyser  * Gerry Hamel, geh@ti.com, Texas Instruments
578acc472SPeter Tyser  */
678acc472SPeter Tyser 
778acc472SPeter Tyser #include <common.h>
878acc472SPeter Tyser #include <malloc.h>
978acc472SPeter Tyser 
1078acc472SPeter Tyser #include <circbuf.h>
1178acc472SPeter Tyser 
1278acc472SPeter Tyser 
buf_init(circbuf_t * buf,unsigned int size)1378acc472SPeter Tyser int buf_init (circbuf_t * buf, unsigned int size)
1478acc472SPeter Tyser {
1578acc472SPeter Tyser 	assert (buf != NULL);
1678acc472SPeter Tyser 
1778acc472SPeter Tyser 	buf->size = 0;
1878acc472SPeter Tyser 	buf->totalsize = size;
1978acc472SPeter Tyser 	buf->data = (char *) malloc (sizeof (char) * size);
2078acc472SPeter Tyser 	assert (buf->data != NULL);
2178acc472SPeter Tyser 
2278acc472SPeter Tyser 	buf->top = buf->data;
2378acc472SPeter Tyser 	buf->tail = buf->data;
2478acc472SPeter Tyser 	buf->end = &(buf->data[size]);
2578acc472SPeter Tyser 
2678acc472SPeter Tyser 	return 1;
2778acc472SPeter Tyser }
2878acc472SPeter Tyser 
buf_free(circbuf_t * buf)2978acc472SPeter Tyser int buf_free (circbuf_t * buf)
3078acc472SPeter Tyser {
3178acc472SPeter Tyser 	assert (buf != NULL);
3278acc472SPeter Tyser 	assert (buf->data != NULL);
3378acc472SPeter Tyser 
3478acc472SPeter Tyser 	free (buf->data);
3578acc472SPeter Tyser 	memset (buf, 0, sizeof (circbuf_t));
3678acc472SPeter Tyser 
3778acc472SPeter Tyser 	return 1;
3878acc472SPeter Tyser }
3978acc472SPeter Tyser 
buf_pop(circbuf_t * buf,char * dest,unsigned int len)4078acc472SPeter Tyser int buf_pop (circbuf_t * buf, char *dest, unsigned int len)
4178acc472SPeter Tyser {
4278acc472SPeter Tyser 	unsigned int i;
4305d887b4Sxypron.glpk@gmx.de 	char *p;
4478acc472SPeter Tyser 
4578acc472SPeter Tyser 	assert (buf != NULL);
4678acc472SPeter Tyser 	assert (dest != NULL);
4778acc472SPeter Tyser 
4805d887b4Sxypron.glpk@gmx.de 	p = buf->top;
4905d887b4Sxypron.glpk@gmx.de 
5078acc472SPeter Tyser 	/* Cap to number of bytes in buffer */
5178acc472SPeter Tyser 	if (len > buf->size)
5278acc472SPeter Tyser 		len = buf->size;
5378acc472SPeter Tyser 
5478acc472SPeter Tyser 	for (i = 0; i < len; i++) {
5578acc472SPeter Tyser 		dest[i] = *p++;
5678acc472SPeter Tyser 		/* Bounds check. */
5778acc472SPeter Tyser 		if (p == buf->end) {
5878acc472SPeter Tyser 			p = buf->data;
5978acc472SPeter Tyser 		}
6078acc472SPeter Tyser 	}
6178acc472SPeter Tyser 
6278acc472SPeter Tyser 	/* Update 'top' pointer */
6378acc472SPeter Tyser 	buf->top = p;
6478acc472SPeter Tyser 	buf->size -= len;
6578acc472SPeter Tyser 
6678acc472SPeter Tyser 	return len;
6778acc472SPeter Tyser }
6878acc472SPeter Tyser 
buf_push(circbuf_t * buf,const char * src,unsigned int len)6978acc472SPeter Tyser int buf_push (circbuf_t * buf, const char *src, unsigned int len)
7078acc472SPeter Tyser {
7178acc472SPeter Tyser 	/* NOTE:  this function allows push to overwrite old data. */
7278acc472SPeter Tyser 	unsigned int i;
7305d887b4Sxypron.glpk@gmx.de 	char *p;
7478acc472SPeter Tyser 
7578acc472SPeter Tyser 	assert (buf != NULL);
7678acc472SPeter Tyser 	assert (src != NULL);
7778acc472SPeter Tyser 
7805d887b4Sxypron.glpk@gmx.de 	p = buf->tail;
7905d887b4Sxypron.glpk@gmx.de 
8078acc472SPeter Tyser 	for (i = 0; i < len; i++) {
8178acc472SPeter Tyser 		*p++ = src[i];
8278acc472SPeter Tyser 		if (p == buf->end) {
8378acc472SPeter Tyser 			p = buf->data;
8478acc472SPeter Tyser 		}
8578acc472SPeter Tyser 		/* Make sure pushing too much data just replaces old data */
8678acc472SPeter Tyser 		if (buf->size < buf->totalsize) {
8778acc472SPeter Tyser 			buf->size++;
8878acc472SPeter Tyser 		} else {
8978acc472SPeter Tyser 			buf->top++;
9078acc472SPeter Tyser 			if (buf->top == buf->end) {
9178acc472SPeter Tyser 				buf->top = buf->data;
9278acc472SPeter Tyser 			}
9378acc472SPeter Tyser 		}
9478acc472SPeter Tyser 	}
9578acc472SPeter Tyser 
9678acc472SPeter Tyser 	/* Update 'tail' pointer */
9778acc472SPeter Tyser 	buf->tail = p;
9878acc472SPeter Tyser 
9978acc472SPeter Tyser 	return len;
10078acc472SPeter Tyser }
101