mirror of
https://github.com/hathach/tinyusb.git
synced 2025-01-17 05:32:55 +08:00
1512 lines
42 KiB
C
1512 lines
42 KiB
C
|
/**
|
||
|
* This header was automatically built using
|
||
|
* embedded_cli.h and embedded_cli.c
|
||
|
* @date 2022-11-03
|
||
|
*
|
||
|
* MIT License
|
||
|
*
|
||
|
* Copyright (c) 2021 Sviatoslav Kokurin (funbiscuit)
|
||
|
*
|
||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
|
* of this software and associated documentation files (the "Software"), to deal
|
||
|
* in the Software without restriction, including without limitation the rights
|
||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
|
* copies of the Software, and to permit persons to whom the Software is
|
||
|
* furnished to do so, subject to the following conditions:
|
||
|
*
|
||
|
* The above copyright notice and this permission notice shall be included in all
|
||
|
* copies or substantial portions of the Software.
|
||
|
*
|
||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||
|
* SOFTWARE.
|
||
|
*/
|
||
|
#ifndef EMBEDDED_CLI_H
|
||
|
#define EMBEDDED_CLI_H
|
||
|
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
|
||
|
extern "C" {
|
||
|
#else
|
||
|
|
||
|
#include <stdbool.h>
|
||
|
|
||
|
#endif
|
||
|
|
||
|
// cstdint is available only since C++11, so use C header
|
||
|
#include <stdint.h>
|
||
|
|
||
|
// used for proper alignment of cli buffer
|
||
|
#if UINTPTR_MAX == 0xFFFF
|
||
|
#define CLI_UINT uint16_t
|
||
|
#elif UINTPTR_MAX == 0xFFFFFFFF
|
||
|
#define CLI_UINT uint32_t
|
||
|
#elif UINTPTR_MAX == 0xFFFFFFFFFFFFFFFFu
|
||
|
#define CLI_UINT uint64_t
|
||
|
#else
|
||
|
#error unsupported pointer size
|
||
|
#endif
|
||
|
|
||
|
#define CLI_UINT_SIZE (sizeof(CLI_UINT))
|
||
|
// convert size in bytes to size in terms of CLI_UINTs (rounded up
|
||
|
// if bytes is not divisible by size of single CLI_UINT)
|
||
|
#define BYTES_TO_CLI_UINTS(bytes) \
|
||
|
(((bytes) + CLI_UINT_SIZE - 1)/CLI_UINT_SIZE)
|
||
|
|
||
|
typedef struct CliCommand CliCommand;
|
||
|
typedef struct CliCommandBinding CliCommandBinding;
|
||
|
typedef struct EmbeddedCli EmbeddedCli;
|
||
|
typedef struct EmbeddedCliConfig EmbeddedCliConfig;
|
||
|
|
||
|
|
||
|
struct CliCommand {
|
||
|
/**
|
||
|
* Name of the command.
|
||
|
* In command "set led 1 1" "set" is name
|
||
|
*/
|
||
|
const char *name;
|
||
|
|
||
|
/**
|
||
|
* String of arguments of the command.
|
||
|
* In command "set led 1 1" "led 1 1" is string of arguments
|
||
|
* Is ended with double 0x00 char
|
||
|
* Use tokenize functions to easily get individual tokens
|
||
|
*/
|
||
|
char *args;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Struct to describe binding of command to function and
|
||
|
*/
|
||
|
struct CliCommandBinding {
|
||
|
/**
|
||
|
* Name of command to bind. Should not be NULL.
|
||
|
*/
|
||
|
const char *name;
|
||
|
|
||
|
/**
|
||
|
* Help string that will be displayed when "help <cmd>" is executed.
|
||
|
* Can have multiple lines separated with "\r\n"
|
||
|
* Can be NULL if no help is provided.
|
||
|
*/
|
||
|
const char *help;
|
||
|
|
||
|
/**
|
||
|
* Flag to perform tokenization before calling binding function.
|
||
|
*/
|
||
|
bool tokenizeArgs;
|
||
|
|
||
|
/**
|
||
|
* Pointer to any specific app context that is required for this binding.
|
||
|
* It will be provided in binding callback.
|
||
|
*/
|
||
|
void *context;
|
||
|
|
||
|
/**
|
||
|
* Binding function for when command is received.
|
||
|
* If null, default callback (onCommand) will be called.
|
||
|
* @param cli - pointer to cli that is calling this binding
|
||
|
* @param args - string of args (if tokenizeArgs is false) or tokens otherwise
|
||
|
* @param context
|
||
|
*/
|
||
|
void (*binding)(EmbeddedCli *cli, char *args, void *context);
|
||
|
};
|
||
|
|
||
|
struct EmbeddedCli {
|
||
|
/**
|
||
|
* Should write char to connection
|
||
|
* @param cli - pointer to cli that executed this function
|
||
|
* @param c - actual character to write
|
||
|
*/
|
||
|
void (*writeChar)(EmbeddedCli *cli, char c);
|
||
|
|
||
|
/**
|
||
|
* Called when command is received and command not found in list of
|
||
|
* command bindings (or binding function is null).
|
||
|
* @param cli - pointer to cli that executed this function
|
||
|
* @param command - pointer to received command
|
||
|
*/
|
||
|
void (*onCommand)(EmbeddedCli *cli, CliCommand *command);
|
||
|
|
||
|
/**
|
||
|
* Can be used by for any application context
|
||
|
*/
|
||
|
void *appContext;
|
||
|
|
||
|
/**
|
||
|
* Pointer to actual implementation, do not use.
|
||
|
*/
|
||
|
void *_impl;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Configuration to create CLI
|
||
|
*/
|
||
|
struct EmbeddedCliConfig {
|
||
|
/**
|
||
|
* Size of buffer that is used to store characters until they're processed
|
||
|
*/
|
||
|
uint16_t rxBufferSize;
|
||
|
|
||
|
/**
|
||
|
* Size of buffer that is used to store current input that is not yet
|
||
|
* sended as command (return not pressed yet)
|
||
|
*/
|
||
|
uint16_t cmdBufferSize;
|
||
|
|
||
|
/**
|
||
|
* Size of buffer that is used to store previously entered commands
|
||
|
* Only unique commands are stored in buffer. If buffer is smaller than
|
||
|
* entered command (including arguments), command is discarded from history
|
||
|
*/
|
||
|
uint16_t historyBufferSize;
|
||
|
|
||
|
/**
|
||
|
* Maximum amount of bindings that can be added via addBinding function.
|
||
|
* Cli increases takes extra bindings for internal commands:
|
||
|
* - help
|
||
|
*/
|
||
|
uint16_t maxBindingCount;
|
||
|
|
||
|
/**
|
||
|
* Buffer to use for cli and all internal structures. If NULL, memory will
|
||
|
* be allocated dynamically. Otherwise this buffer is used and no
|
||
|
* allocations are made
|
||
|
*/
|
||
|
CLI_UINT *cliBuffer;
|
||
|
|
||
|
/**
|
||
|
* Size of buffer for cli and internal structures (in bytes).
|
||
|
*/
|
||
|
uint16_t cliBufferSize;
|
||
|
|
||
|
/**
|
||
|
* Whether autocompletion should be enabled.
|
||
|
* If false, autocompletion is disabled but you still can use 'tab' to
|
||
|
* complete current command manually.
|
||
|
*/
|
||
|
bool enableAutoComplete;
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Returns pointer to default configuration for cli creation. It is safe to
|
||
|
* modify it and then send to embeddedCliNew().
|
||
|
* Returned structure is always the same so do not free and try to use it
|
||
|
* immediately.
|
||
|
* Default values:
|
||
|
* <ul>
|
||
|
* <li>rxBufferSize = 64</li>
|
||
|
* <li>cmdBufferSize = 64</li>
|
||
|
* <li>historyBufferSize = 128</li>
|
||
|
* <li>cliBuffer = NULL (use dynamic allocation)</li>
|
||
|
* <li>cliBufferSize = 0</li>
|
||
|
* <li>maxBindingCount = 8</li>
|
||
|
* <li>enableAutoComplete = true</li>
|
||
|
* </ul>
|
||
|
* @return configuration for cli creation
|
||
|
*/
|
||
|
EmbeddedCliConfig *embeddedCliDefaultConfig(void);
|
||
|
|
||
|
/**
|
||
|
* Returns how many space in config buffer is required for cli creation
|
||
|
* If you provide buffer with less space, embeddedCliNew will return NULL
|
||
|
* This amount will always be divisible by CLI_UINT_SIZE so allocated buffer
|
||
|
* and internal structures can be properly aligned
|
||
|
* @param config
|
||
|
* @return
|
||
|
*/
|
||
|
uint16_t embeddedCliRequiredSize(EmbeddedCliConfig *config);
|
||
|
|
||
|
/**
|
||
|
* Create new CLI.
|
||
|
* Memory is allocated dynamically if cliBuffer in config is NULL.
|
||
|
* After CLI is created, override function pointers to start using it
|
||
|
* @param config - config for cli creation
|
||
|
* @return pointer to created CLI
|
||
|
*/
|
||
|
EmbeddedCli *embeddedCliNew(EmbeddedCliConfig *config);
|
||
|
|
||
|
/**
|
||
|
* Same as calling embeddedCliNew with default config.
|
||
|
* @return
|
||
|
*/
|
||
|
EmbeddedCli *embeddedCliNewDefault(void);
|
||
|
|
||
|
/**
|
||
|
* Receive character and put it to internal buffer
|
||
|
* Actual processing is done inside embeddedCliProcess
|
||
|
* You can call this function from something like interrupt service routine,
|
||
|
* just make sure that you call it only from single place. Otherwise input
|
||
|
* might get corrupted
|
||
|
* @param cli
|
||
|
* @param c - received char
|
||
|
*/
|
||
|
void embeddedCliReceiveChar(EmbeddedCli *cli, char c);
|
||
|
|
||
|
/**
|
||
|
* Process rx/tx buffers. Command callbacks are called from here
|
||
|
* @param cli
|
||
|
*/
|
||
|
void embeddedCliProcess(EmbeddedCli *cli);
|
||
|
|
||
|
/**
|
||
|
* Add specified binding to list of bindings. If list is already full, binding
|
||
|
* is not added and false is returned
|
||
|
* @param cli
|
||
|
* @param binding
|
||
|
* @return true if binding was added, false otherwise
|
||
|
*/
|
||
|
bool embeddedCliAddBinding(EmbeddedCli *cli, CliCommandBinding binding);
|
||
|
|
||
|
/**
|
||
|
* Print specified string and account for currently entered but not submitted
|
||
|
* command.
|
||
|
* Current command is deleted, provided string is printed (with new line) after
|
||
|
* that current command is printed again, so user can continue typing it.
|
||
|
* @param cli
|
||
|
* @param string
|
||
|
*/
|
||
|
void embeddedCliPrint(EmbeddedCli *cli, const char *string);
|
||
|
|
||
|
/**
|
||
|
* Free allocated for cli memory
|
||
|
* @param cli
|
||
|
*/
|
||
|
void embeddedCliFree(EmbeddedCli *cli);
|
||
|
|
||
|
/**
|
||
|
* Perform tokenization of arguments string. Original string is modified and
|
||
|
* should not be used directly (only inside other token functions).
|
||
|
* Individual tokens are separated by single 0x00 char, double 0x00 is put at
|
||
|
* the end of token list. After calling this function, you can use other
|
||
|
* token functions to get individual tokens and token count.
|
||
|
*
|
||
|
* Important: Call this function only once. Otherwise information will be lost if
|
||
|
* more than one token existed
|
||
|
* @param args - string to tokenize (must have extra writable char after 0x00)
|
||
|
* @return
|
||
|
*/
|
||
|
void embeddedCliTokenizeArgs(char *args);
|
||
|
|
||
|
/**
|
||
|
* Return specific token from tokenized string
|
||
|
* @param tokenizedStr
|
||
|
* @param pos (counted from 1)
|
||
|
* @return token
|
||
|
*/
|
||
|
const char *embeddedCliGetToken(const char *tokenizedStr, uint16_t pos);
|
||
|
|
||
|
/**
|
||
|
* Same as embeddedCliGetToken but works on non-const buffer
|
||
|
* @param tokenizedStr
|
||
|
* @param pos (counted from 1)
|
||
|
* @return token
|
||
|
*/
|
||
|
char *embeddedCliGetTokenVariable(char *tokenizedStr, uint16_t pos);
|
||
|
|
||
|
/**
|
||
|
* Find token in provided tokens string and return its position (counted from 1)
|
||
|
* If no such token is found - 0 is returned.
|
||
|
* @param tokenizedStr
|
||
|
* @param token - token to find
|
||
|
* @return position (increased by 1) or zero if no such token found
|
||
|
*/
|
||
|
uint16_t embeddedCliFindToken(const char *tokenizedStr, const char *token);
|
||
|
|
||
|
/**
|
||
|
* Return number of tokens in tokenized string
|
||
|
* @param tokenizedStr
|
||
|
* @return number of tokens
|
||
|
*/
|
||
|
uint16_t embeddedCliGetTokenCount(const char *tokenizedStr);
|
||
|
|
||
|
#ifdef __cplusplus
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
|
||
|
#endif //EMBEDDED_CLI_H
|
||
|
|
||
|
|
||
|
#ifdef EMBEDDED_CLI_IMPL
|
||
|
#ifndef EMBEDDED_CLI_IMPL_GUARD
|
||
|
#define EMBEDDED_CLI_IMPL_GUARD
|
||
|
#ifdef __cplusplus
|
||
|
extern "C" {
|
||
|
#endif
|
||
|
#include <stdlib.h>
|
||
|
#include <string.h>
|
||
|
|
||
|
|
||
|
#define CLI_TOKEN_NPOS 0xffff
|
||
|
|
||
|
#define UNUSED(x) (void)x
|
||
|
|
||
|
#define PREPARE_IMPL(t) \
|
||
|
EmbeddedCliImpl* impl = (EmbeddedCliImpl*)t->_impl
|
||
|
|
||
|
#define IS_FLAG_SET(flags, flag) (((flags) & (flag)) != 0)
|
||
|
|
||
|
#define SET_FLAG(flags, flag) ((flags) |= (flag))
|
||
|
|
||
|
#define UNSET_U8FLAG(flags, flag) ((flags) &= (uint8_t) ~(flag))
|
||
|
|
||
|
/**
|
||
|
* Marks binding as candidate for autocompletion
|
||
|
* This flag is updated each time getAutocompletedCommand is called
|
||
|
*/
|
||
|
#define BINDING_FLAG_AUTOCOMPLETE 1u
|
||
|
|
||
|
/**
|
||
|
* Indicates that rx buffer overflow happened. In such case last command
|
||
|
* that wasn't finished (no \r or \n were received) will be discarded
|
||
|
*/
|
||
|
#define CLI_FLAG_OVERFLOW 0x01u
|
||
|
|
||
|
/**
|
||
|
* Indicates that initialization is completed. Initialization is completed in
|
||
|
* first call to process and needed, for example, to print invitation message.
|
||
|
*/
|
||
|
#define CLI_FLAG_INIT_COMPLETE 0x02u
|
||
|
|
||
|
/**
|
||
|
* Indicates that CLI structure and internal structures were allocated with
|
||
|
* malloc and should bre freed
|
||
|
*/
|
||
|
#define CLI_FLAG_ALLOCATED 0x04u
|
||
|
|
||
|
/**
|
||
|
* Indicates that CLI structure and internal structures were allocated with
|
||
|
* malloc and should bre freed
|
||
|
*/
|
||
|
#define CLI_FLAG_ESCAPE_MODE 0x08u
|
||
|
|
||
|
/**
|
||
|
* Indicates that CLI in mode when it will print directly to output without
|
||
|
* clear of current command and printing it back
|
||
|
*/
|
||
|
#define CLI_FLAG_DIRECT_PRINT 0x10u
|
||
|
|
||
|
/**
|
||
|
* Indicates that live autocompletion is enabled
|
||
|
*/
|
||
|
#define CLI_FLAG_AUTOCOMPLETE_ENABLED 0x20u
|
||
|
|
||
|
typedef struct EmbeddedCliImpl EmbeddedCliImpl;
|
||
|
typedef struct AutocompletedCommand AutocompletedCommand;
|
||
|
typedef struct FifoBuf FifoBuf;
|
||
|
typedef struct CliHistory CliHistory;
|
||
|
|
||
|
struct FifoBuf {
|
||
|
char *buf;
|
||
|
/**
|
||
|
* Position of first element in buffer. From this position elements are taken
|
||
|
*/
|
||
|
uint16_t front;
|
||
|
/**
|
||
|
* Position after last element. At this position new elements are inserted
|
||
|
*/
|
||
|
uint16_t back;
|
||
|
/**
|
||
|
* Size of buffer
|
||
|
*/
|
||
|
uint16_t size;
|
||
|
};
|
||
|
|
||
|
struct CliHistory {
|
||
|
/**
|
||
|
* Items in buffer are separated by null-chars
|
||
|
*/
|
||
|
char *buf;
|
||
|
|
||
|
/**
|
||
|
* Total size of buffer
|
||
|
*/
|
||
|
uint16_t bufferSize;
|
||
|
|
||
|
/**
|
||
|
* Index of currently selected element. This allows to navigate history
|
||
|
* After command is sent, current element is reset to 0 (no element)
|
||
|
*/
|
||
|
uint16_t current;
|
||
|
|
||
|
/**
|
||
|
* Number of items in buffer
|
||
|
* Items are counted from top to bottom (and are 1 based).
|
||
|
* So the most recent item is 1 and the oldest is itemCount.
|
||
|
*/
|
||
|
uint16_t itemsCount;
|
||
|
};
|
||
|
|
||
|
struct EmbeddedCliImpl {
|
||
|
/**
|
||
|
* Invitation string. Is printed at the beginning of each line with user
|
||
|
* input
|
||
|
*/
|
||
|
const char *invitation;
|
||
|
|
||
|
CliHistory history;
|
||
|
|
||
|
/**
|
||
|
* Buffer for storing received chars.
|
||
|
* Chars are stored in FIFO mode.
|
||
|
*/
|
||
|
FifoBuf rxBuffer;
|
||
|
|
||
|
/**
|
||
|
* Buffer for current command
|
||
|
*/
|
||
|
char *cmdBuffer;
|
||
|
|
||
|
/**
|
||
|
* Size of current command
|
||
|
*/
|
||
|
uint16_t cmdSize;
|
||
|
|
||
|
/**
|
||
|
* Total size of command buffer
|
||
|
*/
|
||
|
uint16_t cmdMaxSize;
|
||
|
|
||
|
CliCommandBinding *bindings;
|
||
|
|
||
|
/**
|
||
|
* Flags for each binding. Sizes are the same as for bindings array
|
||
|
*/
|
||
|
uint8_t *bindingsFlags;
|
||
|
|
||
|
uint16_t bindingsCount;
|
||
|
|
||
|
uint16_t maxBindingsCount;
|
||
|
|
||
|
/**
|
||
|
* Total length of input line. This doesn't include invitation but
|
||
|
* includes current command and its live autocompletion
|
||
|
*/
|
||
|
uint16_t inputLineLength;
|
||
|
|
||
|
/**
|
||
|
* Stores last character that was processed.
|
||
|
*/
|
||
|
char lastChar;
|
||
|
|
||
|
/**
|
||
|
* Flags are defined as CLI_FLAG_*
|
||
|
*/
|
||
|
uint8_t flags;
|
||
|
};
|
||
|
|
||
|
struct AutocompletedCommand {
|
||
|
/**
|
||
|
* Name of autocompleted command (or first candidate for autocompletion if
|
||
|
* there are multiple candidates).
|
||
|
* NULL if autocomplete not possible.
|
||
|
*/
|
||
|
const char *firstCandidate;
|
||
|
|
||
|
/**
|
||
|
* Number of characters that can be completed safely. For example, if there
|
||
|
* are two possible commands "get-led" and "get-adc", then for prefix "g"
|
||
|
* autocompletedLen will be 4. If there are only one candidate, this number
|
||
|
* is always equal to length of the command.
|
||
|
*/
|
||
|
uint16_t autocompletedLen;
|
||
|
|
||
|
/**
|
||
|
* Total number of candidates for autocompletion
|
||
|
*/
|
||
|
uint16_t candidateCount;
|
||
|
};
|
||
|
|
||
|
static EmbeddedCliConfig defaultConfig;
|
||
|
|
||
|
/**
|
||
|
* Number of commands that cli adds. Commands:
|
||
|
* - help
|
||
|
*/
|
||
|
static const uint16_t cliInternalBindingCount = 1;
|
||
|
|
||
|
static const char *lineBreak = "\r\n";
|
||
|
|
||
|
/**
|
||
|
* Navigate through command history back and forth. If navigateUp is true,
|
||
|
* navigate to older commands, otherwise navigate to newer.
|
||
|
* When history end is reached, nothing happens.
|
||
|
* @param cli
|
||
|
* @param navigateUp
|
||
|
*/
|
||
|
static void navigateHistory(EmbeddedCli *cli, bool navigateUp);
|
||
|
|
||
|
/**
|
||
|
* Process escaped character. After receiving ESC+[ sequence, all chars up to
|
||
|
* ending character are sent to this function
|
||
|
* @param cli
|
||
|
* @param c
|
||
|
*/
|
||
|
static void onEscapedInput(EmbeddedCli *cli, char c);
|
||
|
|
||
|
/**
|
||
|
* Process input character. Character is valid displayable char and should be
|
||
|
* added to current command string and displayed to client.
|
||
|
* @param cli
|
||
|
* @param c
|
||
|
*/
|
||
|
static void onCharInput(EmbeddedCli *cli, char c);
|
||
|
|
||
|
/**
|
||
|
* Process control character (like \r or \n) possibly altering state of current
|
||
|
* command or executing onCommand callback.
|
||
|
* @param cli
|
||
|
* @param c
|
||
|
*/
|
||
|
static void onControlInput(EmbeddedCli *cli, char c);
|
||
|
|
||
|
/**
|
||
|
* Parse command in buffer and execute callback
|
||
|
* @param cli
|
||
|
*/
|
||
|
static void parseCommand(EmbeddedCli *cli);
|
||
|
|
||
|
/**
|
||
|
* Setup bindings for internal commands, like help
|
||
|
* @param cli
|
||
|
*/
|
||
|
static void initInternalBindings(EmbeddedCli *cli);
|
||
|
|
||
|
/**
|
||
|
* Show help for given tokens (or default help if no tokens)
|
||
|
* @param cli
|
||
|
* @param tokens
|
||
|
* @param context - not used
|
||
|
*/
|
||
|
static void onHelp(EmbeddedCli *cli, char *tokens, void *context);
|
||
|
|
||
|
/**
|
||
|
* Show error about unknown command
|
||
|
* @param cli
|
||
|
* @param name
|
||
|
*/
|
||
|
static void onUnknownCommand(EmbeddedCli *cli, const char *name);
|
||
|
|
||
|
/**
|
||
|
* Return autocompleted command for given prefix.
|
||
|
* Prefix is compared to all known command bindings and autocompleted result
|
||
|
* is returned
|
||
|
* @param cli
|
||
|
* @param prefix
|
||
|
* @return
|
||
|
*/
|
||
|
static AutocompletedCommand getAutocompletedCommand(EmbeddedCli *cli, const char *prefix);
|
||
|
|
||
|
/**
|
||
|
* Prints autocompletion result while keeping current command unchanged
|
||
|
* Prints only if autocompletion is present and only one candidate exists.
|
||
|
* @param cli
|
||
|
*/
|
||
|
static void printLiveAutocompletion(EmbeddedCli *cli);
|
||
|
|
||
|
/**
|
||
|
* Handles autocomplete request. If autocomplete possible - fills current
|
||
|
* command with autocompleted command. When multiple commands satisfy entered
|
||
|
* prefix, they are printed to output.
|
||
|
* @param cli
|
||
|
*/
|
||
|
static void onAutocompleteRequest(EmbeddedCli *cli);
|
||
|
|
||
|
/**
|
||
|
* Removes all input from current line (replaces it with whitespaces)
|
||
|
* And places cursor at the beginning of the line
|
||
|
* @param cli
|
||
|
*/
|
||
|
static void clearCurrentLine(EmbeddedCli *cli);
|
||
|
|
||
|
/**
|
||
|
* Write given string to cli output
|
||
|
* @param cli
|
||
|
* @param str
|
||
|
*/
|
||
|
static void writeToOutput(EmbeddedCli *cli, const char *str);
|
||
|
|
||
|
/**
|
||
|
* Returns true if provided char is a supported control char:
|
||
|
* \r, \n, \b or 0x7F (treated as \b)
|
||
|
* @param c
|
||
|
* @return
|
||
|
*/
|
||
|
static bool isControlChar(char c);
|
||
|
|
||
|
/**
|
||
|
* Returns true if provided char is a valid displayable character:
|
||
|
* a-z, A-Z, 0-9, whitespace, punctuation, etc.
|
||
|
* Currently only ASCII is supported
|
||
|
* @param c
|
||
|
* @return
|
||
|
*/
|
||
|
static bool isDisplayableChar(char c);
|
||
|
|
||
|
/**
|
||
|
* How many elements are currently available in buffer
|
||
|
* @param buffer
|
||
|
* @return number of elements
|
||
|
*/
|
||
|
static uint16_t fifoBufAvailable(FifoBuf *buffer);
|
||
|
|
||
|
/**
|
||
|
* Return first character from buffer and remove it from buffer
|
||
|
* Buffer must be non-empty, otherwise 0 is returned
|
||
|
* @param buffer
|
||
|
* @return
|
||
|
*/
|
||
|
static char fifoBufPop(FifoBuf *buffer);
|
||
|
|
||
|
/**
|
||
|
* Push character into fifo buffer. If there is no space left, character is
|
||
|
* discarded and false is returned
|
||
|
* @param buffer
|
||
|
* @param a - character to add
|
||
|
* @return true if char was added to buffer, false otherwise
|
||
|
*/
|
||
|
static bool fifoBufPush(FifoBuf *buffer, char a);
|
||
|
|
||
|
/**
|
||
|
* Copy provided string to the history buffer.
|
||
|
* If it is already inside history, it will be removed from it and added again.
|
||
|
* So after addition, it will always be on top
|
||
|
* If available size is not enough (and total size is enough) old elements will
|
||
|
* be removed from history so this item can be put to it
|
||
|
* @param history
|
||
|
* @param str
|
||
|
* @return true if string was put in history
|
||
|
*/
|
||
|
static bool historyPut(CliHistory *history, const char *str);
|
||
|
|
||
|
/**
|
||
|
* Get item from history. Items are counted from 1 so if item is 0 or greater
|
||
|
* than itemCount, NULL is returned
|
||
|
* @param history
|
||
|
* @param item
|
||
|
* @return true if string was put in history
|
||
|
*/
|
||
|
static const char *historyGet(CliHistory *history, uint16_t item);
|
||
|
|
||
|
/**
|
||
|
* Remove specific item from history
|
||
|
* @param history
|
||
|
* @param str - string to remove
|
||
|
* @return
|
||
|
*/
|
||
|
static void historyRemove(CliHistory *history, const char *str);
|
||
|
|
||
|
/**
|
||
|
* Return position (index of first char) of specified token
|
||
|
* @param tokenizedStr - tokenized string (separated by \0 with
|
||
|
* \0\0 at the end)
|
||
|
* @param pos - token position (counted from 1)
|
||
|
* @return index of first char of specified token
|
||
|
*/
|
||
|
static uint16_t getTokenPosition(const char *tokenizedStr, uint16_t pos);
|
||
|
|
||
|
EmbeddedCliConfig *embeddedCliDefaultConfig(void) {
|
||
|
defaultConfig.rxBufferSize = 64;
|
||
|
defaultConfig.cmdBufferSize = 64;
|
||
|
defaultConfig.historyBufferSize = 128;
|
||
|
defaultConfig.cliBuffer = NULL;
|
||
|
defaultConfig.cliBufferSize = 0;
|
||
|
defaultConfig.maxBindingCount = 8;
|
||
|
defaultConfig.enableAutoComplete = true;
|
||
|
return &defaultConfig;
|
||
|
}
|
||
|
|
||
|
uint16_t embeddedCliRequiredSize(EmbeddedCliConfig *config) {
|
||
|
uint16_t bindingCount = (uint16_t) (config->maxBindingCount + cliInternalBindingCount);
|
||
|
return (uint16_t) (CLI_UINT_SIZE * (
|
||
|
BYTES_TO_CLI_UINTS(sizeof(EmbeddedCli)) +
|
||
|
BYTES_TO_CLI_UINTS(sizeof(EmbeddedCliImpl)) +
|
||
|
BYTES_TO_CLI_UINTS(config->rxBufferSize * sizeof(char)) +
|
||
|
BYTES_TO_CLI_UINTS(config->cmdBufferSize * sizeof(char)) +
|
||
|
BYTES_TO_CLI_UINTS(config->historyBufferSize * sizeof(char)) +
|
||
|
BYTES_TO_CLI_UINTS(bindingCount * sizeof(CliCommandBinding)) +
|
||
|
BYTES_TO_CLI_UINTS(bindingCount * sizeof(uint8_t))));
|
||
|
}
|
||
|
|
||
|
EmbeddedCli *embeddedCliNew(EmbeddedCliConfig *config) {
|
||
|
EmbeddedCli *cli = NULL;
|
||
|
|
||
|
uint16_t bindingCount = (uint16_t) (config->maxBindingCount + cliInternalBindingCount);
|
||
|
|
||
|
size_t totalSize = embeddedCliRequiredSize(config);
|
||
|
|
||
|
bool allocated = false;
|
||
|
if (config->cliBuffer == NULL) {
|
||
|
config->cliBuffer = (CLI_UINT *) malloc(totalSize); // malloc guarantees alignment.
|
||
|
if (config->cliBuffer == NULL)
|
||
|
return NULL;
|
||
|
allocated = true;
|
||
|
} else if (config->cliBufferSize < totalSize) {
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
CLI_UINT *buf = config->cliBuffer;
|
||
|
|
||
|
memset(buf, 0, totalSize);
|
||
|
|
||
|
cli = (EmbeddedCli *) buf;
|
||
|
buf += BYTES_TO_CLI_UINTS(sizeof(EmbeddedCli));
|
||
|
|
||
|
cli->_impl = (EmbeddedCliImpl *) buf;
|
||
|
buf += BYTES_TO_CLI_UINTS(sizeof(EmbeddedCliImpl));
|
||
|
|
||
|
PREPARE_IMPL(cli);
|
||
|
impl->rxBuffer.buf = (char *) buf;
|
||
|
buf += BYTES_TO_CLI_UINTS(config->rxBufferSize * sizeof(char));
|
||
|
|
||
|
impl->cmdBuffer = (char *) buf;
|
||
|
buf += BYTES_TO_CLI_UINTS(config->cmdBufferSize * sizeof(char));
|
||
|
|
||
|
impl->bindings = (CliCommandBinding *) buf;
|
||
|
buf += BYTES_TO_CLI_UINTS(bindingCount * sizeof(CliCommandBinding));
|
||
|
|
||
|
impl->bindingsFlags = (uint8_t *) buf;
|
||
|
buf += BYTES_TO_CLI_UINTS(bindingCount);
|
||
|
|
||
|
impl->history.buf = (char *) buf;
|
||
|
impl->history.bufferSize = config->historyBufferSize;
|
||
|
|
||
|
if (allocated)
|
||
|
SET_FLAG(impl->flags, CLI_FLAG_ALLOCATED);
|
||
|
|
||
|
if (config->enableAutoComplete)
|
||
|
SET_FLAG(impl->flags, CLI_FLAG_AUTOCOMPLETE_ENABLED);
|
||
|
|
||
|
impl->rxBuffer.size = config->rxBufferSize;
|
||
|
impl->rxBuffer.front = 0;
|
||
|
impl->rxBuffer.back = 0;
|
||
|
impl->cmdMaxSize = config->cmdBufferSize;
|
||
|
impl->bindingsCount = 0;
|
||
|
impl->maxBindingsCount = (uint16_t) (config->maxBindingCount + cliInternalBindingCount);
|
||
|
impl->lastChar = '\0';
|
||
|
impl->invitation = "> ";
|
||
|
|
||
|
initInternalBindings(cli);
|
||
|
|
||
|
return cli;
|
||
|
}
|
||
|
|
||
|
EmbeddedCli *embeddedCliNewDefault(void) {
|
||
|
return embeddedCliNew(embeddedCliDefaultConfig());
|
||
|
}
|
||
|
|
||
|
void embeddedCliReceiveChar(EmbeddedCli *cli, char c) {
|
||
|
PREPARE_IMPL(cli);
|
||
|
|
||
|
if (!fifoBufPush(&impl->rxBuffer, c)) {
|
||
|
SET_FLAG(impl->flags, CLI_FLAG_OVERFLOW);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void embeddedCliProcess(EmbeddedCli *cli) {
|
||
|
if (cli->writeChar == NULL)
|
||
|
return;
|
||
|
|
||
|
PREPARE_IMPL(cli);
|
||
|
|
||
|
|
||
|
if (!IS_FLAG_SET(impl->flags, CLI_FLAG_INIT_COMPLETE)) {
|
||
|
SET_FLAG(impl->flags, CLI_FLAG_INIT_COMPLETE);
|
||
|
writeToOutput(cli, impl->invitation);
|
||
|
}
|
||
|
|
||
|
while (fifoBufAvailable(&impl->rxBuffer)) {
|
||
|
char c = fifoBufPop(&impl->rxBuffer);
|
||
|
|
||
|
if (IS_FLAG_SET(impl->flags, CLI_FLAG_ESCAPE_MODE)) {
|
||
|
onEscapedInput(cli, c);
|
||
|
} else if (impl->lastChar == 0x1B && c == '[') {
|
||
|
//enter escape mode
|
||
|
SET_FLAG(impl->flags, CLI_FLAG_ESCAPE_MODE);
|
||
|
} else if (isControlChar(c)) {
|
||
|
onControlInput(cli, c);
|
||
|
} else if (isDisplayableChar(c)) {
|
||
|
onCharInput(cli, c);
|
||
|
}
|
||
|
|
||
|
printLiveAutocompletion(cli);
|
||
|
|
||
|
impl->lastChar = c;
|
||
|
}
|
||
|
|
||
|
// discard unfinished command if overflow happened
|
||
|
if (IS_FLAG_SET(impl->flags, CLI_FLAG_OVERFLOW)) {
|
||
|
impl->cmdSize = 0;
|
||
|
impl->cmdBuffer[impl->cmdSize] = '\0';
|
||
|
UNSET_U8FLAG(impl->flags, CLI_FLAG_OVERFLOW);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
bool embeddedCliAddBinding(EmbeddedCli *cli, CliCommandBinding binding) {
|
||
|
PREPARE_IMPL(cli);
|
||
|
if (impl->bindingsCount == impl->maxBindingsCount)
|
||
|
return false;
|
||
|
|
||
|
impl->bindings[impl->bindingsCount] = binding;
|
||
|
|
||
|
++impl->bindingsCount;
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
void embeddedCliPrint(EmbeddedCli *cli, const char *string) {
|
||
|
if (cli->writeChar == NULL)
|
||
|
return;
|
||
|
|
||
|
PREPARE_IMPL(cli);
|
||
|
|
||
|
// remove chars for autocompletion and live command
|
||
|
if (!IS_FLAG_SET(impl->flags, CLI_FLAG_DIRECT_PRINT))
|
||
|
clearCurrentLine(cli);
|
||
|
|
||
|
// print provided string
|
||
|
writeToOutput(cli, string);
|
||
|
writeToOutput(cli, lineBreak);
|
||
|
|
||
|
// print current command back to screen
|
||
|
if (!IS_FLAG_SET(impl->flags, CLI_FLAG_DIRECT_PRINT)) {
|
||
|
writeToOutput(cli, impl->invitation);
|
||
|
writeToOutput(cli, impl->cmdBuffer);
|
||
|
impl->inputLineLength = impl->cmdSize;
|
||
|
|
||
|
printLiveAutocompletion(cli);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void embeddedCliFree(EmbeddedCli *cli) {
|
||
|
PREPARE_IMPL(cli);
|
||
|
if (IS_FLAG_SET(impl->flags, CLI_FLAG_ALLOCATED)) {
|
||
|
// allocation is done in single call to malloc, so need only single free
|
||
|
free(cli);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void embeddedCliTokenizeArgs(char *args) {
|
||
|
if (args == NULL)
|
||
|
return;
|
||
|
|
||
|
// for now only space, but can add more later
|
||
|
const char *separators = " ";
|
||
|
|
||
|
// indicates that arg is quoted so separators are copied as is
|
||
|
bool quotesEnabled = false;
|
||
|
// indicates that previous char was a slash, so next char is copied as is
|
||
|
bool escapeActivated = false;
|
||
|
int insertPos = 0;
|
||
|
|
||
|
int i = 0;
|
||
|
char currentChar;
|
||
|
while ((currentChar = args[i]) != '\0') {
|
||
|
++i;
|
||
|
|
||
|
if (escapeActivated) {
|
||
|
escapeActivated = false;
|
||
|
} else if (currentChar == '\\') {
|
||
|
escapeActivated = true;
|
||
|
continue;
|
||
|
} else if (currentChar == '"') {
|
||
|
quotesEnabled = !quotesEnabled;
|
||
|
currentChar = '\0';
|
||
|
} else if (!quotesEnabled && strchr(separators, currentChar) != NULL) {
|
||
|
currentChar = '\0';
|
||
|
}
|
||
|
|
||
|
// null chars are only copied once and not copied to the beginning
|
||
|
if (currentChar != '\0' || (insertPos > 0 && args[insertPos - 1] != '\0')) {
|
||
|
args[insertPos] = currentChar;
|
||
|
++insertPos;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// make args double null-terminated source buffer must be big enough to contain extra spaces
|
||
|
args[insertPos] = '\0';
|
||
|
args[insertPos + 1] = '\0';
|
||
|
}
|
||
|
|
||
|
const char *embeddedCliGetToken(const char *tokenizedStr, uint16_t pos) {
|
||
|
uint16_t i = getTokenPosition(tokenizedStr, pos);
|
||
|
|
||
|
if (i != CLI_TOKEN_NPOS)
|
||
|
return &tokenizedStr[i];
|
||
|
else
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
char *embeddedCliGetTokenVariable(char *tokenizedStr, uint16_t pos) {
|
||
|
uint16_t i = getTokenPosition(tokenizedStr, pos);
|
||
|
|
||
|
if (i != CLI_TOKEN_NPOS)
|
||
|
return &tokenizedStr[i];
|
||
|
else
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
uint16_t embeddedCliFindToken(const char *tokenizedStr, const char *token) {
|
||
|
if (tokenizedStr == NULL || token == NULL)
|
||
|
return 0;
|
||
|
|
||
|
uint16_t size = embeddedCliGetTokenCount(tokenizedStr);
|
||
|
for (uint16_t i = 1; i <= size; ++i) {
|
||
|
if (strcmp(embeddedCliGetToken(tokenizedStr, i), token) == 0)
|
||
|
return i;
|
||
|
}
|
||
|
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
uint16_t embeddedCliGetTokenCount(const char *tokenizedStr) {
|
||
|
if (tokenizedStr == NULL || tokenizedStr[0] == '\0')
|
||
|
return 0;
|
||
|
|
||
|
int i = 0;
|
||
|
uint16_t tokenCount = 1;
|
||
|
while (true) {
|
||
|
if (tokenizedStr[i] == '\0') {
|
||
|
if (tokenizedStr[i + 1] == '\0')
|
||
|
break;
|
||
|
++tokenCount;
|
||
|
}
|
||
|
++i;
|
||
|
}
|
||
|
|
||
|
return tokenCount;
|
||
|
}
|
||
|
|
||
|
static void navigateHistory(EmbeddedCli *cli, bool navigateUp) {
|
||
|
PREPARE_IMPL(cli);
|
||
|
if (impl->history.itemsCount == 0 ||
|
||
|
(navigateUp && impl->history.current == impl->history.itemsCount) ||
|
||
|
(!navigateUp && impl->history.current == 0))
|
||
|
return;
|
||
|
|
||
|
clearCurrentLine(cli);
|
||
|
|
||
|
writeToOutput(cli, impl->invitation);
|
||
|
|
||
|
if (navigateUp)
|
||
|
++impl->history.current;
|
||
|
else
|
||
|
--impl->history.current;
|
||
|
|
||
|
const char *item = historyGet(&impl->history, impl->history.current);
|
||
|
// simple way to handle empty command the same way as others
|
||
|
if (item == NULL)
|
||
|
item = "";
|
||
|
uint16_t len = (uint16_t) strlen(item);
|
||
|
memcpy(impl->cmdBuffer, item, len);
|
||
|
impl->cmdBuffer[len] = '\0';
|
||
|
impl->cmdSize = len;
|
||
|
|
||
|
writeToOutput(cli, impl->cmdBuffer);
|
||
|
impl->inputLineLength = impl->cmdSize;
|
||
|
|
||
|
printLiveAutocompletion(cli);
|
||
|
}
|
||
|
|
||
|
static void onEscapedInput(EmbeddedCli *cli, char c) {
|
||
|
PREPARE_IMPL(cli);
|
||
|
|
||
|
if (c >= 64 && c <= 126) {
|
||
|
// handle escape sequence
|
||
|
UNSET_U8FLAG(impl->flags, CLI_FLAG_ESCAPE_MODE);
|
||
|
|
||
|
if (c == 'A' || c == 'B') {
|
||
|
// treat \e[..A as cursor up and \e[..B as cursor down
|
||
|
// there might be extra chars between [ and A/B, just ignore them
|
||
|
navigateHistory(cli, c == 'A');
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void onCharInput(EmbeddedCli *cli, char c) {
|
||
|
PREPARE_IMPL(cli);
|
||
|
|
||
|
// have to reserve two extra chars for command ending (used in tokenization)
|
||
|
if (impl->cmdSize + 2 >= impl->cmdMaxSize)
|
||
|
return;
|
||
|
|
||
|
impl->cmdBuffer[impl->cmdSize] = c;
|
||
|
++impl->cmdSize;
|
||
|
impl->cmdBuffer[impl->cmdSize] = '\0';
|
||
|
|
||
|
cli->writeChar(cli, c);
|
||
|
}
|
||
|
|
||
|
static void onControlInput(EmbeddedCli *cli, char c) {
|
||
|
PREPARE_IMPL(cli);
|
||
|
|
||
|
// process \r\n and \n\r as single \r\n command
|
||
|
if ((impl->lastChar == '\r' && c == '\n') ||
|
||
|
(impl->lastChar == '\n' && c == '\r'))
|
||
|
return;
|
||
|
|
||
|
if (c == '\r' || c == '\n') {
|
||
|
// try to autocomplete command and then process it
|
||
|
onAutocompleteRequest(cli);
|
||
|
|
||
|
writeToOutput(cli, lineBreak);
|
||
|
|
||
|
if (impl->cmdSize > 0)
|
||
|
parseCommand(cli);
|
||
|
impl->cmdSize = 0;
|
||
|
impl->cmdBuffer[impl->cmdSize] = '\0';
|
||
|
impl->inputLineLength = 0;
|
||
|
impl->history.current = 0;
|
||
|
|
||
|
writeToOutput(cli, impl->invitation);
|
||
|
} else if ((c == '\b' || c == 0x7F) && impl->cmdSize > 0) {
|
||
|
// remove char from screen
|
||
|
cli->writeChar(cli, '\b');
|
||
|
cli->writeChar(cli, ' ');
|
||
|
cli->writeChar(cli, '\b');
|
||
|
// and from buffer
|
||
|
--impl->cmdSize;
|
||
|
impl->cmdBuffer[impl->cmdSize] = '\0';
|
||
|
} else if (c == '\t') {
|
||
|
onAutocompleteRequest(cli);
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|
||
|
static void parseCommand(EmbeddedCli *cli) {
|
||
|
PREPARE_IMPL(cli);
|
||
|
|
||
|
bool isEmpty = true;
|
||
|
|
||
|
for (int i = 0; i < impl->cmdSize; ++i) {
|
||
|
if (impl->cmdBuffer[i] != ' ') {
|
||
|
isEmpty = false;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
// do not process empty commands
|
||
|
if (isEmpty)
|
||
|
return;
|
||
|
// push command to history before buffer is modified
|
||
|
historyPut(&impl->history, impl->cmdBuffer);
|
||
|
|
||
|
char *cmdName = NULL;
|
||
|
char *cmdArgs = NULL;
|
||
|
bool nameFinished = false;
|
||
|
|
||
|
// find command name and command args inside command buffer
|
||
|
for (int i = 0; i < impl->cmdSize; ++i) {
|
||
|
char c = impl->cmdBuffer[i];
|
||
|
|
||
|
if (c == ' ') {
|
||
|
// all spaces between name and args are filled with zeros
|
||
|
// so name is a correct null-terminated string
|
||
|
if (cmdArgs == NULL)
|
||
|
impl->cmdBuffer[i] = '\0';
|
||
|
if (cmdName != NULL)
|
||
|
nameFinished = true;
|
||
|
|
||
|
} else if (cmdName == NULL) {
|
||
|
cmdName = &impl->cmdBuffer[i];
|
||
|
} else if (cmdArgs == NULL && nameFinished) {
|
||
|
cmdArgs = &impl->cmdBuffer[i];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// we keep two last bytes in cmd buffer reserved so cmdSize is always by 2
|
||
|
// less than cmdMaxSize
|
||
|
impl->cmdBuffer[impl->cmdSize + 1] = '\0';
|
||
|
|
||
|
if (cmdName == NULL)
|
||
|
return;
|
||
|
|
||
|
// try to find command in bindings
|
||
|
for (int i = 0; i < impl->bindingsCount; ++i) {
|
||
|
if (strcmp(cmdName, impl->bindings[i].name) == 0) {
|
||
|
if (impl->bindings[i].binding == NULL)
|
||
|
break;
|
||
|
|
||
|
if (impl->bindings[i].tokenizeArgs)
|
||
|
embeddedCliTokenizeArgs(cmdArgs);
|
||
|
// currently, output is blank line, so we can just print directly
|
||
|
SET_FLAG(impl->flags, CLI_FLAG_DIRECT_PRINT);
|
||
|
impl->bindings[i].binding(cli, cmdArgs, impl->bindings[i].context);
|
||
|
UNSET_U8FLAG(impl->flags, CLI_FLAG_DIRECT_PRINT);
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// command not found in bindings or binding was null
|
||
|
// try to call default callback
|
||
|
if (cli->onCommand != NULL) {
|
||
|
CliCommand command;
|
||
|
command.name = cmdName;
|
||
|
command.args = cmdArgs;
|
||
|
|
||
|
// currently, output is blank line, so we can just print directly
|
||
|
SET_FLAG(impl->flags, CLI_FLAG_DIRECT_PRINT);
|
||
|
cli->onCommand(cli, &command);
|
||
|
UNSET_U8FLAG(impl->flags, CLI_FLAG_DIRECT_PRINT);
|
||
|
} else {
|
||
|
onUnknownCommand(cli, cmdName);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void initInternalBindings(EmbeddedCli *cli) {
|
||
|
CliCommandBinding b = {
|
||
|
"help",
|
||
|
"Print list of commands",
|
||
|
true,
|
||
|
NULL,
|
||
|
onHelp
|
||
|
};
|
||
|
embeddedCliAddBinding(cli, b);
|
||
|
}
|
||
|
|
||
|
static void onHelp(EmbeddedCli *cli, char *tokens, void *context) {
|
||
|
UNUSED(context);
|
||
|
PREPARE_IMPL(cli);
|
||
|
|
||
|
if (impl->bindingsCount == 0) {
|
||
|
writeToOutput(cli, "Help is not available");
|
||
|
writeToOutput(cli, lineBreak);
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
uint16_t tokenCount = embeddedCliGetTokenCount(tokens);
|
||
|
if (tokenCount == 0) {
|
||
|
for (int i = 0; i < impl->bindingsCount; ++i) {
|
||
|
writeToOutput(cli, " * ");
|
||
|
writeToOutput(cli, impl->bindings[i].name);
|
||
|
writeToOutput(cli, lineBreak);
|
||
|
if (impl->bindings[i].help != NULL) {
|
||
|
cli->writeChar(cli, '\t');
|
||
|
writeToOutput(cli, impl->bindings[i].help);
|
||
|
writeToOutput(cli, lineBreak);
|
||
|
}
|
||
|
}
|
||
|
} else if (tokenCount == 1) {
|
||
|
// try find command
|
||
|
const char *helpStr = NULL;
|
||
|
const char *cmdName = embeddedCliGetToken(tokens, 1);
|
||
|
bool found = false;
|
||
|
for (int i = 0; i < impl->bindingsCount; ++i) {
|
||
|
if (strcmp(impl->bindings[i].name, cmdName) == 0) {
|
||
|
helpStr = impl->bindings[i].help;
|
||
|
found = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (found && helpStr != NULL) {
|
||
|
writeToOutput(cli, " * ");
|
||
|
writeToOutput(cli, cmdName);
|
||
|
writeToOutput(cli, lineBreak);
|
||
|
cli->writeChar(cli, '\t');
|
||
|
writeToOutput(cli, helpStr);
|
||
|
writeToOutput(cli, lineBreak);
|
||
|
} else if (found) {
|
||
|
writeToOutput(cli, "Help is not available");
|
||
|
writeToOutput(cli, lineBreak);
|
||
|
} else {
|
||
|
onUnknownCommand(cli, cmdName);
|
||
|
}
|
||
|
} else {
|
||
|
writeToOutput(cli, "Command \"help\" receives one or zero arguments");
|
||
|
writeToOutput(cli, lineBreak);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static void onUnknownCommand(EmbeddedCli *cli, const char *name) {
|
||
|
writeToOutput(cli, "Unknown command: \"");
|
||
|
writeToOutput(cli, name);
|
||
|
writeToOutput(cli, "\". Write \"help\" for a list of available commands");
|
||
|
writeToOutput(cli, lineBreak);
|
||
|
}
|
||
|
|
||
|
static AutocompletedCommand getAutocompletedCommand(EmbeddedCli *cli, const char *prefix) {
|
||
|
AutocompletedCommand cmd = {NULL, 0, 0};
|
||
|
|
||
|
size_t prefixLen = strlen(prefix);
|
||
|
|
||
|
PREPARE_IMPL(cli);
|
||
|
if (impl->bindingsCount == 0 || prefixLen == 0)
|
||
|
return cmd;
|
||
|
|
||
|
|
||
|
for (int i = 0; i < impl->bindingsCount; ++i) {
|
||
|
const char *name = impl->bindings[i].name;
|
||
|
size_t len = strlen(name);
|
||
|
|
||
|
// unset autocomplete flag
|
||
|
UNSET_U8FLAG(impl->bindingsFlags[i], BINDING_FLAG_AUTOCOMPLETE);
|
||
|
|
||
|
if (len < prefixLen)
|
||
|
continue;
|
||
|
|
||
|
// check if this command is candidate for autocomplete
|
||
|
bool isCandidate = true;
|
||
|
for (size_t j = 0; j < prefixLen; ++j) {
|
||
|
if (prefix[j] != name[j]) {
|
||
|
isCandidate = false;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!isCandidate)
|
||
|
continue;
|
||
|
|
||
|
impl->bindingsFlags[i] |= BINDING_FLAG_AUTOCOMPLETE;
|
||
|
|
||
|
if (cmd.candidateCount == 0 || len < cmd.autocompletedLen)
|
||
|
cmd.autocompletedLen = (uint16_t) len;
|
||
|
|
||
|
++cmd.candidateCount;
|
||
|
|
||
|
if (cmd.candidateCount == 1) {
|
||
|
cmd.firstCandidate = name;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
for (size_t j = impl->cmdSize; j < cmd.autocompletedLen; ++j) {
|
||
|
if (cmd.firstCandidate[j] != name[j]) {
|
||
|
cmd.autocompletedLen = (uint16_t) j;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return cmd;
|
||
|
}
|
||
|
|
||
|
static void printLiveAutocompletion(EmbeddedCli *cli) {
|
||
|
PREPARE_IMPL(cli);
|
||
|
|
||
|
if (!IS_FLAG_SET(impl->flags, CLI_FLAG_AUTOCOMPLETE_ENABLED))
|
||
|
return;
|
||
|
|
||
|
AutocompletedCommand cmd = getAutocompletedCommand(cli, impl->cmdBuffer);
|
||
|
|
||
|
if (cmd.candidateCount == 0) {
|
||
|
cmd.autocompletedLen = impl->cmdSize;
|
||
|
}
|
||
|
|
||
|
// print live autocompletion (or nothing, if it doesn't exist)
|
||
|
for (size_t i = impl->cmdSize; i < cmd.autocompletedLen; ++i) {
|
||
|
cli->writeChar(cli, cmd.firstCandidate[i]);
|
||
|
}
|
||
|
// replace with spaces previous autocompletion
|
||
|
for (size_t i = cmd.autocompletedLen; i < impl->inputLineLength; ++i) {
|
||
|
cli->writeChar(cli, ' ');
|
||
|
}
|
||
|
impl->inputLineLength = cmd.autocompletedLen;
|
||
|
cli->writeChar(cli, '\r');
|
||
|
// print current command again so cursor is moved to initial place
|
||
|
writeToOutput(cli, impl->invitation);
|
||
|
writeToOutput(cli, impl->cmdBuffer);
|
||
|
}
|
||
|
|
||
|
static void onAutocompleteRequest(EmbeddedCli *cli) {
|
||
|
PREPARE_IMPL(cli);
|
||
|
|
||
|
AutocompletedCommand cmd = getAutocompletedCommand(cli, impl->cmdBuffer);
|
||
|
|
||
|
if (cmd.candidateCount == 0)
|
||
|
return;
|
||
|
|
||
|
if (cmd.candidateCount == 1 || cmd.autocompletedLen > impl->cmdSize) {
|
||
|
// can copy from index cmdSize, but prefix is the same, so copy everything
|
||
|
memcpy(impl->cmdBuffer, cmd.firstCandidate, cmd.autocompletedLen);
|
||
|
if (cmd.candidateCount == 1) {
|
||
|
impl->cmdBuffer[cmd.autocompletedLen] = ' ';
|
||
|
++cmd.autocompletedLen;
|
||
|
}
|
||
|
impl->cmdBuffer[cmd.autocompletedLen] = '\0';
|
||
|
|
||
|
writeToOutput(cli, &impl->cmdBuffer[impl->cmdSize]);
|
||
|
impl->cmdSize = cmd.autocompletedLen;
|
||
|
impl->inputLineLength = impl->cmdSize;
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
// with multiple candidates when we already completed to common prefix
|
||
|
// we show all candidates and print input again
|
||
|
// we need to completely clear current line since it begins with invitation
|
||
|
clearCurrentLine(cli);
|
||
|
|
||
|
for (int i = 0; i < impl->bindingsCount; ++i) {
|
||
|
// autocomplete flag is set for all candidates by last call to
|
||
|
// getAutocompletedCommand
|
||
|
if (!(impl->bindingsFlags[i] & BINDING_FLAG_AUTOCOMPLETE))
|
||
|
continue;
|
||
|
|
||
|
const char *name = impl->bindings[i].name;
|
||
|
|
||
|
writeToOutput(cli, name);
|
||
|
writeToOutput(cli, lineBreak);
|
||
|
}
|
||
|
|
||
|
writeToOutput(cli, impl->invitation);
|
||
|
writeToOutput(cli, impl->cmdBuffer);
|
||
|
|
||
|
impl->inputLineLength = impl->cmdSize;
|
||
|
}
|
||
|
|
||
|
static void clearCurrentLine(EmbeddedCli *cli) {
|
||
|
PREPARE_IMPL(cli);
|
||
|
size_t len = impl->inputLineLength + strlen(impl->invitation);
|
||
|
|
||
|
cli->writeChar(cli, '\r');
|
||
|
for (size_t i = 0; i < len; ++i) {
|
||
|
cli->writeChar(cli, ' ');
|
||
|
}
|
||
|
cli->writeChar(cli, '\r');
|
||
|
impl->inputLineLength = 0;
|
||
|
}
|
||
|
|
||
|
static void writeToOutput(EmbeddedCli *cli, const char *str) {
|
||
|
size_t len = strlen(str);
|
||
|
|
||
|
for (size_t i = 0; i < len; ++i) {
|
||
|
cli->writeChar(cli, str[i]);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static bool isControlChar(char c) {
|
||
|
return c == '\r' || c == '\n' || c == '\b' || c == '\t' || c == 0x7F;
|
||
|
}
|
||
|
|
||
|
static bool isDisplayableChar(char c) {
|
||
|
return (c >= 32 && c <= 126);
|
||
|
}
|
||
|
|
||
|
static uint16_t fifoBufAvailable(FifoBuf *buffer) {
|
||
|
if (buffer->back >= buffer->front)
|
||
|
return (uint16_t) (buffer->back - buffer->front);
|
||
|
else
|
||
|
return (uint16_t) (buffer->size - buffer->front + buffer->back);
|
||
|
}
|
||
|
|
||
|
static char fifoBufPop(FifoBuf *buffer) {
|
||
|
char a = '\0';
|
||
|
if (buffer->front != buffer->back) {
|
||
|
a = buffer->buf[buffer->front];
|
||
|
buffer->front = (uint16_t) (buffer->front + 1) % buffer->size;
|
||
|
}
|
||
|
return a;
|
||
|
}
|
||
|
|
||
|
static bool fifoBufPush(FifoBuf *buffer, char a) {
|
||
|
uint16_t newBack = (uint16_t) (buffer->back + 1) % buffer->size;
|
||
|
if (newBack != buffer->front) {
|
||
|
buffer->buf[buffer->back] = a;
|
||
|
buffer->back = newBack;
|
||
|
return true;
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
static bool historyPut(CliHistory *history, const char *str) {
|
||
|
size_t len = strlen(str);
|
||
|
// each item is ended with \0 so, need to have that much space at least
|
||
|
if (history->bufferSize < len + 1)
|
||
|
return false;
|
||
|
|
||
|
// remove str from history (if it's present) so we don't get duplicates
|
||
|
historyRemove(history, str);
|
||
|
|
||
|
size_t usedSize;
|
||
|
// remove old items if new one can't fit into buffer
|
||
|
while (history->itemsCount > 0) {
|
||
|
const char *item = historyGet(history, history->itemsCount);
|
||
|
size_t itemLen = strlen(item);
|
||
|
usedSize = ((size_t) (item - history->buf)) + itemLen + 1;
|
||
|
|
||
|
size_t freeSpace = history->bufferSize - usedSize;
|
||
|
|
||
|
if (freeSpace >= len + 1)
|
||
|
break;
|
||
|
|
||
|
// space not enough, remove last element
|
||
|
--history->itemsCount;
|
||
|
}
|
||
|
if (history->itemsCount > 0) {
|
||
|
// when history not empty, shift elements so new item is first
|
||
|
memmove(&history->buf[len + 1], history->buf, usedSize);
|
||
|
}
|
||
|
memcpy(history->buf, str, len + 1);
|
||
|
++history->itemsCount;
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
static const char *historyGet(CliHistory *history, uint16_t item) {
|
||
|
if (item == 0 || item > history->itemsCount)
|
||
|
return NULL;
|
||
|
|
||
|
// items are stored in the same way (separated by \0 and counted from 1),
|
||
|
// so can use this call
|
||
|
return embeddedCliGetToken(history->buf, item);
|
||
|
}
|
||
|
|
||
|
static void historyRemove(CliHistory *history, const char *str) {
|
||
|
if (str == NULL || history->itemsCount == 0)
|
||
|
return;
|
||
|
char *item = NULL;
|
||
|
uint16_t itemPosition;
|
||
|
for (itemPosition = 1; itemPosition <= history->itemsCount; ++itemPosition) {
|
||
|
// items are stored in the same way (separated by \0 and counted from 1),
|
||
|
// so can use this call
|
||
|
item = embeddedCliGetTokenVariable(history->buf, itemPosition);
|
||
|
if (strcmp(item, str) == 0) {
|
||
|
break;
|
||
|
}
|
||
|
item = NULL;
|
||
|
}
|
||
|
if (item == NULL)
|
||
|
return;
|
||
|
|
||
|
--history->itemsCount;
|
||
|
if (itemPosition == (history->itemsCount + 1)) {
|
||
|
// if this is a last element, nothing is remaining to move
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
size_t len = strlen(item);
|
||
|
size_t remaining = (size_t) (history->bufferSize - (item + len + 1 - history->buf));
|
||
|
// move everything to the right of found item
|
||
|
memmove(item, &item[len + 1], remaining);
|
||
|
}
|
||
|
|
||
|
static uint16_t getTokenPosition(const char *tokenizedStr, uint16_t pos) {
|
||
|
if (tokenizedStr == NULL || pos == 0)
|
||
|
return CLI_TOKEN_NPOS;
|
||
|
uint16_t i = 0;
|
||
|
uint16_t tokenCount = 1;
|
||
|
while (true) {
|
||
|
if (tokenCount == pos)
|
||
|
break;
|
||
|
|
||
|
if (tokenizedStr[i] == '\0') {
|
||
|
++tokenCount;
|
||
|
if (tokenizedStr[i + 1] == '\0')
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
++i;
|
||
|
}
|
||
|
|
||
|
if (tokenizedStr[i] != '\0')
|
||
|
return i;
|
||
|
else
|
||
|
return CLI_TOKEN_NPOS;
|
||
|
}
|
||
|
#ifdef __cplusplus
|
||
|
}
|
||
|
#endif
|
||
|
#endif // EMBEDDED_CLI_IMPL_GUARD
|
||
|
#endif // EMBEDDED_CLI_IMPL
|