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