This repository has been archived by the owner on Aug 22, 2024. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathSerialCommand.cpp
166 lines (148 loc) · 4.96 KB
/
SerialCommand.cpp
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
/**
SerialCommand - A Wiring/Arduino library to tokenize and parse commands
received over a serial port.
Copyright (C) 2021 Csaba Balazs
Copyright (C) 2012 Stefan Rado
Copyright (C) 2011 Steven Cogswell <[email protected]>
http://husks.wordpress.com
Version 20210320
This library is free software: you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This 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 Lesser General Public License for more details.
You should have received a copy of the GNU General Public License
along with this library. If not, see <http://www.gnu.org/licenses/>.
*/
#include "SerialCommand.h"
/**
Constructor makes sure some things are set.
*/
SerialCommand::SerialCommand()
: commandCount(0),
defaultHandler(NULL),
nullHandler(NULL),
term('\n'),
term2('\r'),
last(NULL)
{
strcpy(delim, " "); // strtok_r needs a null-terminated string
clearBuffer();
}
/**
Adds a "command" and a handler function to the list of available commands.
This is used for matching a found token in the buffer, and gives the pointer
to the handler function to deal with it.
*/
void SerialCommand::addCommand(const char *command, void (*function)()) {
#ifdef SERIALCOMMAND_DEBUG
Serial.print("Adding command (");
Serial.print(commandCount);
Serial.print("): ");
Serial.println(command);
#endif
if (commandCount >= SERIALCOMMAND_MAXCOMMANDCOUNT)
return;
strncpy(commandList[commandCount].command, command, SERIALCOMMAND_MAXCOMMANDLENGTH);
commandList[commandCount].function = function;
commandCount++;
}
/**
This sets up a handler to be called in the event that the receveived command string
isn't in the list of commands.
*/
void SerialCommand::setDefaultHandler(void (*function)(const char *)) {
defaultHandler = function;
}
/**
This sets up a handler to be called in the event that the user hits enter on a
blank line
*/
void SerialCommand::setNullHandler(void (*function)()) {
nullHandler = function;
}
/**
This checks the Serial stream for characters, and assembles them into a buffer.
When the terminator character (default '\n') is seen, it starts parsing the
buffer for a prefix command, and calls handlers setup by addCommand() member
*/
void SerialCommand::readSerial() {
while (Serial.available() > 0) {
char inChar = Serial.read(); // Read single available character, there may be more waiting
#if defined SERIALCOMMAND_DEBUG || defined SERIAL_COMMAND_ECHO
Serial.print(inChar); // Echo back to serial stream
#endif
if ((inChar == term) || (inChar == term2)) { // Check for the terminator (default '\r') meaning end of command
#ifdef SERIALCOMMAND_DEBUG
Serial.print("Received: ");
Serial.println(buffer);
#endif
char *command = strtok_r(buffer, delim, &last); // Search for command at start of buffer
if (command != NULL) {
bool matched = false;
for (int i = 0; i < commandCount; i++) {
#ifdef SERIALCOMMAND_DEBUG
Serial.print("Comparing [");
Serial.print(command);
Serial.print("] to [");
Serial.print(commandList[i].command);
Serial.println("]");
#endif
// Compare the found command against the list of known commands for a match
if (strncmp(command, commandList[i].command, SERIALCOMMAND_MAXCOMMANDLENGTH) == 0) {
#ifdef SERIALCOMMAND_DEBUG
Serial.print("Matched Command: ");
Serial.println(command);
#endif
// Execute the stored handler function for the command
(*commandList[i].function)();
matched = true;
break;
}
}
if (!matched && (defaultHandler != NULL)) {
(*defaultHandler)(command);
}
}
else {
if (nullHandler)
(*nullHandler)();
}
clearBuffer();
}
else if (isprint(inChar)) { // Only printable characters into the buffer
if (bufPos < SERIALCOMMAND_BUFFER) {
buffer[bufPos++] = inChar; // Put character into buffer
buffer[bufPos] = '\0'; // Null terminate
} else {
#ifdef SERIALCOMMAND_DEBUG
Serial.println("Line buffer is full - increase SERIALCOMMAND_BUFFER");
#endif
}
}
else if (inChar == 127 || inChar == 8) {
if (bufPos)
{
bufPos--;
buffer[bufPos] = '\0';
}
}
}
}
/*
Clear the input buffer.
*/
void SerialCommand::clearBuffer() {
buffer[0] = '\0';
bufPos = 0;
}
/**
Retrieve the next token ("word" or "argument") from the command buffer.
Returns NULL if no more tokens exist.
*/
char *SerialCommand::next() {
return strtok_r(NULL, delim, &last);
}