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
 |