forked from dhansel/Altair8800
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathfifo.h
141 lines (104 loc) · 3.62 KB
/
fifo.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
/**
** soft_uart library
** Copyright (C) 2015-2018
**
** Antonio C. Domínguez Brito <[email protected]>
** División de Robótica y Oceanografía Computacional <www.roc.siani.es>
** and Departamento de Informática y Sistemas <www.dis.ulpgc.es>
** Universidad de Las Palmas de Gran Canaria (ULPGC) <www.ulpgc.es>
**
** This file is part of the soft_uart library.
** The soft_uart library 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 3 of the License, or any
** later version.
**
** The soft_uart library is distributed in the hope that it will be useful,
** but WITHOUT ANY WARRANTY; without even the implied warranty of
** MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
** Public License for more details.
**
** You should have received a copy (COPYING file) of the GNU General Public
** License along with the soft_uart library.
** If not, see: <http://www.gnu.org/licenses/>.
**/
/*
* File: fifo.h
* Description: This is the implementation of a static fifo using templates.
* The fifo can be circular or not
* Date: July 8th, 2015
* Author: Antonio C. Dominguez-Brito <[email protected]>
* ROC-SIANI - Universidad de Las Palmas de Gran Canaria - Spain
*/
#ifndef FIFO_H
#define FIFO_H
#include <cstddef>
namespace arduino_due
{
template<bool BOOL_VALUE>
struct bool_to_type { static constexpr bool value=BOOL_VALUE; };
template<typename T,size_t LENGTH,bool IS_CIRCULAR = false>
class basic_fifo
{
public:
basic_fifo() { _init_(); }
~basic_fifo() {}
bool push(const T& t)
{ return _push_(t,bool_to_type<IS_CIRCULAR>()); }
bool pop(T& t)
{
if(_first_<0) return false;
t=_buffer_p_[_first_];
if(_first_==_last_) _init_();
else { _first_=(_first_+1)%LENGTH; _items_--; }
return true;
}
bool is_empty() { return (_first_<0); }
bool is_full() { return (_items_==LENGTH); }
int items() { return _items_; }
int available() { return LENGTH-_items_; }
void reset() { _init_(); }
private:
T _buffer_p_[LENGTH];
int _first_,_last_;
volatile int _items_;
void _init_() { _first_=_last_=-1; _items_=0; }
// push implementation for a normal fifo
// when is full, we can not push any further element
bool _push_(const T& t,bool_to_type<false>)
{
int new_last=(_last_+1)%LENGTH;
if(new_last==_first_) // full?
return false;
_last_=new_last;
_buffer_p_[_last_]=t;
if(_first_<0) _first_=_last_;
_items_++;
return true;
}
// push implementation for a circular fifo
// when is full, we overwrite the first element
bool _push_(const T& t,bool_to_type<true>)
{
int new_last=(_last_+1)%LENGTH;
if(new_last==_first_) // full?
{
// NOTE: when the fifo is circular and the fifo is full
// the new element overwrites the first one
_first_=(_first_+1)%LENGTH;
_last_=new_last; _buffer_p_[_last_]=t;
return false;
}
_last_=new_last;
_buffer_p_[_last_]=t;
if(_first_<0) _first_=_last_;
_items_++;
return true;
}
};
template<typename T, size_t LENGTH>
using fifo=basic_fifo<T,LENGTH,false>;
template<typename T, size_t LENGTH>
using circular_fifo=basic_fifo<T,LENGTH,true>;
}
#endif // FIFO_H