-
-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathdigitalio.hpp
356 lines (281 loc) · 13.9 KB
/
digitalio.hpp
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
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
// SPDX-License-Identifier: GPL-3.0-or-later
//
// Copyright (c) 2013-2023 plan44.ch / Lukas Zeller, Zurich, Switzerland
//
// Author: Lukas Zeller <luz@plan44.ch>
//
// This file is part of p44utils.
//
// p44utils 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
// (at your option) any later version.
//
// p44utils 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 of the GNU General Public License
// along with p44utils. If not, see <http://www.gnu.org/licenses/>.
//
#ifndef __p44utils__digitalio__
#define __p44utils__digitalio__
#include "p44utils_main.hpp"
#include "iopin.hpp"
#if ENABLE_P44SCRIPT && !defined(ENABLE_DIGITALIO_SCRIPT_FUNCS)
#define ENABLE_DIGITALIO_SCRIPT_FUNCS 1
#endif
#if ENABLE_DIGITALIO_SCRIPT_FUNCS
#include "p44script.hpp"
#endif
using namespace std;
namespace p44 {
/// Generic digital I/O
class DigitalIo :
public P44Obj
#if ENABLE_DIGITALIO_SCRIPT_FUNCS && ENABLE_P44SCRIPT
, public P44Script::EventSource
#endif
{
IOPinPtr mIoPin; ///< the actual hardware interface to the pin
string mPinSpec;
bool mOutput;
bool mInverted;
Tristate mPull; // yes = pull up, no = pull down, undefined = no pull
public:
/// Create general purpose I/O
/// @param aPinSpec specification of the IO; form is [+][-][/][bus.device.]pin,
/// where optional leading plus '+'/'-' enable pullup/down (if device supports it)
/// optional slash '/' inverts the polarity, and where bus & device can be omitted for normal GPIOs.
/// @param aOutput use as output
/// @param aInitialState initial state (to set for output, to expect without triggering change for input)
/// Note: aInitialState is logic state (pin state is inverse if aPinSpec is slash-prefixed)
/// @note possible pin specifications are
/// "missing" : dummy (non-connected) pin
/// "gpio.N" or just "N": standard Linux or ESP32 GPIO number N
/// "led.N": standard Linux LED number N
/// "gpioNS9XXXX.NAME" : DigiESP Linux GPIO named NAME
/// "i2cN.DEVICE@i2caddr.pinNumber" : numbered pin of DEVICE at i2caddr on i2c bus N
/// (DEVICE is name of chip, such as PCF8574 or TCA9555)
/// "spiXY.DEVICE[-options]@spiaddr.pinNumber" : numbered pin of DEVICE at spiaddr on spidevX.Y
/// (DEVICE is name of chip, such as MCP23S17)
/// possible options are:
/// H: use inverted phase (compared to original microwire SPI)
/// P: use inverted polarity (compared to original microwire SPI)
/// C: chip select high
/// N: no chip select
/// 3: 3 wire
/// R: SPI ready, slave pulls low to pause
/// S: slow speed (1/10 of bus' normal speed)
/// s: very slow speed (1/100 of bus' normal speed)
DigitalIo(const char* aPinSpec, bool aOutput, bool aInitialState = false);
virtual ~DigitalIo();
/// get name
string getName() const;
/// check for output
bool isOutput() const { return mOutput; };
/// get state of GPIO
/// @return current state (from actual IO pin for inputs, from last set state for outputs)
bool isSet();
/// set state of output (NOP for inputs)
/// @param aState new state to set output to
void set(bool aState);
/// set state to true
void on();
/// set state to false
void off();
/// toggle state of output and return new state
/// @return new state of output after toggling (for inputs, just returns state like isSet() does)
bool toggle();
/// install state change detector callback
/// @note using this method of change detection will disable event delivery, as enabled by setChangeDetection()
/// @param aInputChangedCB will be called when the input state changes. Passing NULL disables input state change reporting.
/// @param aDebounceTime after a reported state change, next input sampling will take place only after specified interval
/// @param aPollInterval if <0 (Infinite), the state change detector only works if the input pin supports state change
/// detection without polling (e.g. with GPIO edge trigger). If aPollInterval is >=0, and the
/// input pin does not support edge detection, the state detection will be implemented via polling
/// on the current mainloop - if pollInterval==0 then polling will be done with a default interval, otherwise in
/// the specified interval
/// @return true if input supports the type of state change detection requested
virtual bool setInputChangedHandler(InputChangedCB aInputChangedCB, MLMicroSeconds aDebounceTime, MLMicroSeconds aPollInterval);
#if ENABLE_DIGITALIO_SCRIPT_FUNCS && ENABLE_P44SCRIPT
/// setup change detection with delivering changes to registered event sinks
/// @note using this method of change detection will disable change callback, as enabled by setInputChangedHandler()
/// @param aDebounceTime after a reported state change, next input sampling will take place only after specified interval.
/// Passing aDebounceTime to <0 means disabling change detection
/// @param aPollInterval if <0 (Infinite), the state change detector only works if the input pin supports state change
/// detection without polling (e.g. with GPIO edge trigger). If aPollInterval is >=0, and the
/// input pin does not support edge detection, the state detection will be implemented via polling
/// on the current mainloop - if pollInterval==0 then polling will be done with a default interval, otherwise in
/// the specified interval
/// @return true if input supports the type of state change detection requested
bool setChangeDetection(MLMicroSeconds aDebounceTime=-1, MLMicroSeconds aPollInterval=0);
/// get a digital input state object. This is also what is sent to event sinks
P44Script::ScriptObjPtr getStateObj();
private:
void processChange(bool aNewState);
#endif // ENABLE_DIGITALIO_SCRIPT_FUNCS && ENABLE_P44SCRIPT
};
typedef boost::intrusive_ptr<DigitalIo> DigitalIoPtr;
#if !REDUCED_FOOTPRINT
/// bus of multiple digital lines read or written as a number
class DigitalIoBus : public P44Obj
{
typedef std::vector<DigitalIoPtr> BusPinVector;
BusPinVector mBusPins;
bool mOutputs;
uint32_t mCurrentValue;
public:
/// Create bus consiting of multiple general purpose I/O pins
/// @param aBusPinSpecs comma separated list of bus pin specifications, MSB first.
/// `(prefix)` can be used to define multiple similar pins, such as `(gpio.)1,2,3,4,(i2c0.MCP23017@24.)6,7`
/// @param aNumBits (max) number of bits
/// @param aOutputs use as outputs
/// @param aInitialStates initial state of all bus pins
DigitalIoBus(const char* aBusPinSpecs, int aNumBits, bool aOutputs, bool aInitialStates = false);
virtual ~DigitalIoBus();
/// get the current bus value
uint32_t getBusValue();
/// get the max bus value
uint32_t getMaxBusValue();
/// get the bus width in number of bits
uint8_t getBusWidth();
/// set the current bus value
/// @param aBusValue new value
/// @note only actually changes pin values if bit position in aBusValue has changed since last setBusValue()
/// @note actual digital value on outputs will not change synchronously for all pins, but sequentially from LSB to MSB
void setBusValue(uint32_t aBusValue);
};
typedef boost::intrusive_ptr<DigitalIoBus> DigitalIoBusPtr;
#endif // REDUCED_FOOTPRINT
/// GPIO used as pushbutton
class ButtonInput : public DigitalIo
{
typedef DigitalIo inherited;
public:
/// button event handler
/// @param aState the current state of the button (relevant when handler was installed with aPressAndRelease set)
/// @param aHasChanged set when reporting a state change, cleared when reporting the same state again (when repeatActiveReport set)
/// @param aTimeSincePreviousChange time passed since previous button state change (to easily detect long press actions etc.)
typedef boost::function<void (bool aState, bool aHasChanged, MLMicroSeconds aTimeSincePreviousChange)> ButtonHandlerCB;
private:
MLMicroSeconds mLastChangeTime;
bool mReportPressAndRelease;
ButtonHandlerCB mButtonHandler;
MLMicroSeconds mRepeatActiveReport;
MLTicket mActiveReportTicket;
void inputChanged(bool aNewState);
void repeatStateReport();
public:
/// Create pushbutton
/// @param aPinSpec specification of the pin where the pushbutton is connected (can be prefixed with slash to invert, plus to enable pullup)
ButtonInput(const char* aPinSpec);
/// destructor
virtual ~ButtonInput();
/// set handler to be called on pushbutton events
/// @param aButtonHandler handler for pushbutton events
/// @param aPressAndRelease if set, both pressing and releasing button generates event.
/// Otherwise, only one event is issued per button press (on button release)
/// @param aRepeatActiveReport time after which a still pressed button is reported again (to detect long presses without extra timers)
void setButtonHandler(ButtonHandlerCB aButtonHandler, bool aPressAndRelease, MLMicroSeconds aRepeatActiveReport=p44::Never);
};
typedef boost::intrusive_ptr<ButtonInput> ButtonInputPtr;
/// GPIO used for indicator (e.g. LED)
class IndicatorOutput : public DigitalIo
{
typedef DigitalIo inherited;
bool mNextTimedState; // what state should be set next time the timer triggers
MLMicroSeconds mBlinkOnTime;
MLMicroSeconds mBlinkOffTime;
MLMicroSeconds mBlinkUntilTime;
MLTicket mTimedOpTicket;
void timer(MLTimer &aTimer);
public:
/// Create indicator output
/// @param aPinSpec specification of the pin where the pushbutton is connected (can be prefixed with slash to invert)
/// @param aInitiallyOn initial state (on or off) of the indicator
IndicatorOutput(const char* aPinSpec, bool aInitiallyOn = false);
/// destructor
virtual ~IndicatorOutput();
/// activate the output for a certain time period, then switch off again
/// @param aOnTime how long indicator should stay active
void onFor(MLMicroSeconds aOnTime);
/// blink indicator for a certain time period, with a given blink period and on ratio
/// @param aOnTime how long indicator should stay active, or p44::infinite to keep blinking
/// @param aBlinkPeriod how fast the blinking should be
/// @param aOnRatioPercent how many percents of aBlinkPeriod the indicator should be on
void blinkFor(MLMicroSeconds aOnTime, MLMicroSeconds aBlinkPeriod = 600*MilliSecond, int aOnRatioPercent = 50);
/// stop blinking/timed activation immediately, but do not change state of the indicator
void stop();
/// stop blinking/timed activation immediately and turn indicator off
void steady(bool aState);
/// stop blinking/timed activation immediately and turn indicator off
void steadyOff();
/// stop blinking/timed activation immediately and turn indicator on
void steadyOn();
};
typedef boost::intrusive_ptr<IndicatorOutput> IndicatorOutputPtr;
#if ENABLE_DIGITALIO_SCRIPT_FUNCS && ENABLE_P44SCRIPT
namespace P44Script {
class DigitalIoObj;
/// represents a state change from a digital input
class DigitalInputEventObj : public NumericValue
{
typedef NumericValue inherited;
DigitalIoPtr mDigitalIo;
public:
DigitalInputEventObj(DigitalIoPtr aDigitalIo);
virtual void deactivate() P44_OVERRIDE;
virtual string getAnnotation() const P44_OVERRIDE;
virtual TypeInfo getTypeInfo() const P44_OVERRIDE;
virtual bool isEventSource() const P44_OVERRIDE;
virtual void registerForFilteredEvents(EventSink* aEventSink, intptr_t aRegId = 0) P44_OVERRIDE;
};
/// represents a digital I/O
/// @note is an event source, but does not expose it directly, only via DigitalInputEventObjs
class DigitalIoObj : public StructuredLookupObject
{
typedef StructuredLookupObject inherited;
DigitalIoPtr mDigitalIo;
public:
DigitalIoObj(DigitalIoPtr aDigitalIo);
virtual string getAnnotation() const P44_OVERRIDE { return "digitalIO"; };
DigitalIoPtr digitalIo() { return mDigitalIo; }
void inputChanged(bool aNewState);
/// factory method to get a DigitalIo either by creating it from pinspec
/// string or by using existing DigitalIoObj passed
static DigitalIoPtr digitalIoFromArg(ScriptObjPtr aArg, bool aOutput, bool aInitialState);
};
/// represents a digital bus
/// @note is an event source, but does not expose it directly, only via DigitalInputEventObjs
class DigitalIoBusObj : public StructuredLookupObject
{
typedef StructuredLookupObject inherited;
DigitalIoBusPtr mDigitalIoBus;
public:
DigitalIoBusObj(DigitalIoBusPtr aDigitalIoBus);
virtual string getAnnotation() const P44_OVERRIDE { return "digitalBus"; };
DigitalIoBusPtr digitalIoBus() { return mDigitalIoBus; }
};
/// represents a indicator light
class IndicatorObj : public StructuredLookupObject
{
typedef StructuredLookupObject inherited;
IndicatorOutputPtr mIndicator;
public:
IndicatorObj(IndicatorOutputPtr aIndicator);
virtual string getAnnotation() const P44_OVERRIDE { return "indicator"; };
IndicatorOutputPtr indicator() { return mIndicator; }
};
/// represents the global objects related to Digitalio
class DigitalIoLookup : public BuiltInMemberLookup
{
typedef BuiltInMemberLookup inherited;
public:
DigitalIoLookup();
};
}
#endif // ENABLE_DIGITALIO_SCRIPT_FUNCS && ENABLE_P44SCRIPT
} // namespace p44
#endif /* defined(__p44utils__digitalio__) */