| 
									
										
										
										
											2012-08-01 20:26:12 +10:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * Generic FIFO component, implemented as a circular buffer. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Copyright (c) 2012 Peter A. G. Crosthwaite | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This program is free software; you can redistribute it and/or | 
					
						
							|  |  |  |  * modify it under the terms of the GNU General Public License | 
					
						
							|  |  |  |  * as published by the Free Software Foundation; either version | 
					
						
							|  |  |  |  * 2 of the License, or (at your option) any later version. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * You should have received a copy of the GNU General Public License along | 
					
						
							|  |  |  |  * with this program; if not, see <http://www.gnu.org/licenses/>.
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-29 17:49:55 +00:00
										 |  |  | #include "qemu/osdep.h"
 | 
					
						
							| 
									
										
										
										
											2019-08-12 07:23:45 +02:00
										 |  |  | #include "migration/vmstate.h"
 | 
					
						
							| 
									
										
										
										
											2013-02-04 10:57:50 +01:00
										 |  |  | #include "qemu/fifo8.h"
 | 
					
						
							| 
									
										
										
										
											2012-08-01 20:26:12 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  | void fifo8_create(Fifo8 *fifo, uint32_t capacity) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     fifo->data = g_new(uint8_t, capacity); | 
					
						
							|  |  |  |     fifo->capacity = capacity; | 
					
						
							|  |  |  |     fifo->head = 0; | 
					
						
							|  |  |  |     fifo->num = 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void fifo8_destroy(Fifo8 *fifo) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     g_free(fifo->data); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void fifo8_push(Fifo8 *fifo, uint8_t data) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2021-01-28 22:17:27 +00:00
										 |  |  |     assert(fifo->num < fifo->capacity); | 
					
						
							| 
									
										
										
										
											2012-08-01 20:26:12 +10:00
										 |  |  |     fifo->data[(fifo->head + fifo->num) % fifo->capacity] = data; | 
					
						
							|  |  |  |     fifo->num++; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-30 23:02:04 +01:00
										 |  |  | void fifo8_push_all(Fifo8 *fifo, const uint8_t *data, uint32_t num) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint32_t start, avail; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-28 22:17:27 +00:00
										 |  |  |     assert(fifo->num + num <= fifo->capacity); | 
					
						
							| 
									
										
										
										
											2014-01-30 23:02:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     start = (fifo->head + fifo->num) % fifo->capacity; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (start + num <= fifo->capacity) { | 
					
						
							|  |  |  |         memcpy(&fifo->data[start], data, num); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         avail = fifo->capacity - start; | 
					
						
							|  |  |  |         memcpy(&fifo->data[start], data, avail); | 
					
						
							|  |  |  |         memcpy(&fifo->data[0], &data[avail], num - avail); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     fifo->num += num; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-01 20:26:12 +10:00
										 |  |  | uint8_t fifo8_pop(Fifo8 *fifo) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint8_t ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-28 22:17:27 +00:00
										 |  |  |     assert(fifo->num > 0); | 
					
						
							| 
									
										
										
										
											2012-08-01 20:26:12 +10:00
										 |  |  |     ret = fifo->data[fifo->head++]; | 
					
						
							|  |  |  |     fifo->head %= fifo->capacity; | 
					
						
							|  |  |  |     fifo->num--; | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-30 23:02:04 +01:00
										 |  |  | const uint8_t *fifo8_pop_buf(Fifo8 *fifo, uint32_t max, uint32_t *num) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     uint8_t *ret; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-28 22:17:27 +00:00
										 |  |  |     assert(max > 0 && max <= fifo->num); | 
					
						
							| 
									
										
										
										
											2014-01-30 23:02:04 +01:00
										 |  |  |     *num = MIN(fifo->capacity - fifo->head, max); | 
					
						
							|  |  |  |     ret = &fifo->data[fifo->head]; | 
					
						
							|  |  |  |     fifo->head += *num; | 
					
						
							|  |  |  |     fifo->head %= fifo->capacity; | 
					
						
							|  |  |  |     fifo->num -= *num; | 
					
						
							|  |  |  |     return ret; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-01 20:26:12 +10:00
										 |  |  | void fifo8_reset(Fifo8 *fifo) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     fifo->num = 0; | 
					
						
							| 
									
										
										
										
											2014-01-30 23:02:05 +01:00
										 |  |  |     fifo->head = 0; | 
					
						
							| 
									
										
										
										
											2012-08-01 20:26:12 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool fifo8_is_empty(Fifo8 *fifo) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (fifo->num == 0); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool fifo8_is_full(Fifo8 *fifo) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return (fifo->num == fifo->capacity); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-01-30 23:02:04 +01:00
										 |  |  | uint32_t fifo8_num_free(Fifo8 *fifo) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return fifo->capacity - fifo->num; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | uint32_t fifo8_num_used(Fifo8 *fifo) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return fifo->num; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-01 20:26:12 +10:00
										 |  |  | const VMStateDescription vmstate_fifo8 = { | 
					
						
							|  |  |  |     .name = "Fifo8", | 
					
						
							|  |  |  |     .version_id = 1, | 
					
						
							|  |  |  |     .minimum_version_id = 1, | 
					
						
							| 
									
										
										
										
											2014-05-13 16:09:35 +01:00
										 |  |  |     .fields = (VMStateField[]) { | 
					
						
							| 
									
										
										
										
											2017-02-03 18:52:17 +01:00
										 |  |  |         VMSTATE_VBUFFER_UINT32(data, Fifo8, 1, NULL, capacity), | 
					
						
							| 
									
										
										
										
											2012-08-01 20:26:12 +10:00
										 |  |  |         VMSTATE_UINT32(head, Fifo8), | 
					
						
							|  |  |  |         VMSTATE_UINT32(num, Fifo8), | 
					
						
							|  |  |  |         VMSTATE_END_OF_LIST() | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | }; |