nuttx-apps/examples/lvglterm/lvglterm.c
Lee Lup Yuen c57e7a7b81 examples/lvglterm: Add LVGL Terminal for NSH
This PR adds an LVGL App that executes NSH Commands (entered with a Touchscreen Keyboard) and renders the NSH Output. The app follows the same design as the `lvgldemo` app and is explained here: ["NuttX RTOS for PinePhone: LVGL Terminal for NSH Shell"](https://lupyuen.github.io/articles/terminal)

`examples/README.md`: Added doc for `lvglterm` app

`examples/lvglterm/lvglterm.c`: LVGL Terminal App

`examples/lvglterm/Makefile`, `Make.defs`: Makefile for LVGL Terminal

`examples/lvglterm/Kconfig`: Added menuconfig option for "Application Configuration > Examples > LVGL Terminal"
2023-02-07 20:53:24 +08:00

589 lines
15 KiB
C

/****************************************************************************
* apps/examples/lvglterm/lvglterm.c
*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership. The
* ASF licenses this file to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance with the
* License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
* License for the specific language governing permissions and limitations
* under the License.
*
****************************************************************************/
/* Reference:
* "NuttX RTOS for PinePhone: LVGL Terminal for NSH Shell"
* https://lupyuen.github.io/articles/terminal
*/
/****************************************************************************
* Included Files
****************************************************************************/
#include <nuttx/config.h>
#include <sys/boardctl.h>
#include <unistd.h>
#include <stddef.h>
#include <stdlib.h>
#include <stdio.h>
#include <time.h>
#include <debug.h>
#include <poll.h>
#include <spawn.h>
#include <lvgl/lvgl.h>
#include <port/lv_port.h>
/* NSH Task requires posix_spawn() */
#ifndef CONFIG_LIBC_EXECFUNCS
# error posix_spawn() should be enabled in the configuration
#endif
/* NSH Redirection requires Pipes */
#ifndef CONFIG_DEV_PIPE_SIZE
# error FIFO and Named Pipe Drivers should be enabled in the configuration
#endif
/* NSH Output requires a Monospaced Font */
#ifndef CONFIG_LV_FONT_UNSCII_16
# error LVGL Font UNSCII 16 should be enabled in the configuration
#endif
/****************************************************************************
* Pre-processor Definitions
****************************************************************************/
/* Should we perform board-specific driver initialization? There are two
* ways that board initialization can occur: 1) automatically via
* board_late_initialize() during bootupif CONFIG_BOARD_LATE_INITIALIZE
* or 2).
* via a call to boardctl() if the interface is enabled
* (CONFIG_BOARDCTL=y).
* If this task is running as an NSH built-in application, then that
* initialization has probably already been performed otherwise we do it
* here.
*/
#undef NEED_BOARDINIT
#if defined(CONFIG_BOARDCTL) && !defined(CONFIG_NSH_ARCHINIT)
# define NEED_BOARDINIT 1
#endif
/* How often to poll for output from NSH Shell (milliseconds) */
#define TIMER_PERIOD_MS 100
/* Read and Write Pipes for NSH stdin, stdout and stderr */
#define READ_PIPE 0
#define WRITE_PIPE 1
/* NSH Task to be started */
#define NSH_TASK "nsh"
/****************************************************************************
* Private Function Prototypes
****************************************************************************/
static int create_widgets(void);
static void timer_callback(lv_timer_t * timer);
static void input_callback(lv_event_t * e);
static bool has_input(int fd);
static void remove_escape_codes(char *buf, int len);
/****************************************************************************
* Private Data
****************************************************************************/
/* Pipes for NSH Shell: stdin, stdout, stderr */
static int g_nsh_stdin[2];
static int g_nsh_stdout[2];
static int g_nsh_stderr[2];
/* LVGL Column Container for NSH Widgets */
static lv_obj_t *g_col;
/* LVGL Text Area Widgets for NSH Input and Output */
static lv_obj_t *g_input;
static lv_obj_t *g_output;
/* LVGL Keyboard Widget for NSH Terminal */
static lv_obj_t *g_kb;
/* LVGL Font Style for NSH Input and Output */
static lv_style_t g_terminal_style;
/* LVGL Timer for polling NSH Output */
static lv_timer_t *g_timer;
/* Arguments for NSH Task */
static char * const g_nsh_argv[] =
{
NSH_TASK, NULL
};
/****************************************************************************
* Private Functions
****************************************************************************/
/****************************************************************************
* Name: create_terminal
*
* Description:
* Create the LVGL Terminal. Start the NSH Shell and redirect the NSH
* stdin, stdout and stderr to the LVGL Widgets.
*
* Input Parameters:
* None
*
* Returned Value:
* Zero (OK) on success; a negated errno value is returned on any failure.
*
****************************************************************************/
static int create_terminal(void)
{
int ret;
pid_t pid;
/* Create the pipes for NSH Shell: stdin, stdout and stderr */
ret = pipe(g_nsh_stdin);
if (ret < 0)
{
_err("stdin pipe failed: %d\n", errno);
return ERROR;
}
ret = pipe(g_nsh_stdout);
if (ret < 0)
{
_err("stdout pipe failed: %d\n", errno);
return ERROR;
}
ret = pipe(g_nsh_stderr);
if (ret < 0)
{
_err("stderr pipe failed: %d\n", errno);
return ERROR;
}
/* Close default stdin, stdout and stderr */
close(0);
close(1);
close(2);
/* Assign the new pipes as stdin, stdout and stderr */
dup2(g_nsh_stdin[READ_PIPE], 0);
dup2(g_nsh_stdout[WRITE_PIPE], 1);
dup2(g_nsh_stderr[WRITE_PIPE], 2);
/* Start the NSH Shell and inherit stdin, stdout and stderr */
ret = posix_spawn(&pid, /* Returned Task ID */
NSH_TASK, /* NSH Path */
NULL, /* Inherit stdin, stdout and stderr */
NULL, /* Default spawn attributes */
g_nsh_argv, /* Arguments */
NULL); /* No environment */
if (ret < 0)
{
int errcode = errno;
_err("posix_spawn failed: %d\n", errcode);
return -errcode;
}
/* Create an LVGL Timer to poll for output from NSH Shell */
g_timer = lv_timer_create(timer_callback, /* Callback Function */
TIMER_PERIOD_MS, /* Timer Period (millisec) */
NULL); /* Callback Argument */
DEBUGASSERT(g_timer != NULL);
/* Create the LVGL Terminal Widgets */
ret = create_widgets();
if (ret < 0)
{
return ret;
}
return OK;
}
/****************************************************************************
* Name: create_widgets
*
* Description:
* Create the LVGL Widgets for LVGL Terminal.
*
* Input Parameters:
* None
*
* Returned Value:
* Zero (OK) on success; a negated errno value is returned on any failure.
*
****************************************************************************/
static int create_widgets(void)
{
/* Set the Font Style for NSH Input and Output to a Monospaced Font */
lv_style_init(&g_terminal_style);
lv_style_set_text_font(&g_terminal_style, &lv_font_unscii_16);
/* Create an LVGL Container with Column Flex Direction */
g_col = lv_obj_create(lv_scr_act());
DEBUGASSERT(g_col != NULL);
lv_obj_set_size(g_col, LV_PCT(100), LV_PCT(100));
lv_obj_set_flex_flow(g_col, LV_FLEX_FLOW_COLUMN);
lv_obj_set_style_pad_all(g_col, 0, 0); /* No padding */
/* Create an LVGL Text Area Widget for NSH Output */
g_output = lv_textarea_create(g_col);
DEBUGASSERT(g_output != NULL);
lv_obj_add_style(g_output, &g_terminal_style, 0);
lv_obj_set_width(g_output, LV_PCT(100));
lv_obj_set_flex_grow(g_output, 1); /* Fill the column */
/* Create an LVGL Text Area Widget for NSH Input */
g_input = lv_textarea_create(g_col);
DEBUGASSERT(g_input != NULL);
lv_obj_add_style(g_input, &g_terminal_style, 0);
lv_obj_set_size(g_input, LV_PCT(100), LV_SIZE_CONTENT);
/* Create an LVGL Keyboard Widget */
g_kb = lv_keyboard_create(g_col);
DEBUGASSERT(g_kb != NULL);
lv_obj_set_style_pad_all(g_kb, 0, 0); /* No padding */
/* Register the Callback Function for NSH Input */
lv_obj_add_event_cb(g_input, input_callback, LV_EVENT_ALL, NULL);
/* Set the Keyboard to populate the NSH Input Text Area */
lv_keyboard_set_textarea(g_kb, g_input);
return OK;
}
/****************************************************************************
* Name: timer_callback
*
* Description:
* Callback Function for LVGL Timer. Poll NSH stdout and stderr for output
* and display the output.
*
* Input Parameters:
* timer - LVGL Timer for the callback
*
* Returned Value:
* None
*
****************************************************************************/
static void timer_callback(lv_timer_t *timer)
{
int ret;
static char buf[64];
DEBUGASSERT(g_nsh_stdout[READ_PIPE] != 0);
DEBUGASSERT(g_nsh_stderr[READ_PIPE] != 0);
/* Poll NSH stdout to check if there's output to be processed */
if (has_input(g_nsh_stdout[READ_PIPE]))
{
/* Read the output from NSH stdout */
ret = read(g_nsh_stdout[READ_PIPE], buf, sizeof(buf) - 1);
if (ret > 0)
{
/* Add to NSH Output Text Area */
buf[ret] = 0;
remove_escape_codes(buf, ret);
DEBUGASSERT(g_output != NULL);
lv_textarea_add_text(g_output, buf);
}
}
/* Poll NSH stderr to check if there's output to be processed */
if (has_input(g_nsh_stderr[READ_PIPE]))
{
/* Read the output from NSH stderr */
ret = read(g_nsh_stderr[READ_PIPE], buf, sizeof(buf) - 1);
if (ret > 0)
{
/* Add to NSH Output Text Area */
buf[ret] = 0;
remove_escape_codes(buf, ret);
DEBUGASSERT(g_output != NULL);
lv_textarea_add_text(g_output, buf);
}
}
}
/****************************************************************************
* Name: input_callback
*
* Description:
* Callback Function for NSH Input Text Area. If Enter Key was pressed,
* send the NSH Input Command to NSH stdin.
*
* Input Parameters:
* e - LVGL Event for the callback
*
* Returned Value:
* None
*
****************************************************************************/
static void input_callback(lv_event_t *e)
{
int ret;
/* Decode the LVGL Event */
const lv_event_code_t code = lv_event_get_code(e);
/* If NSH Input Text Area has changed, get the Key Pressed */
if (code == LV_EVENT_VALUE_CHANGED)
{
/* Get the Button Index of the Keyboard Button Pressed */
const uint16_t id = lv_keyboard_get_selected_btn(g_kb);
/* Get the Text of the Keyboard Button */
const char *key = lv_keyboard_get_btn_text(g_kb, id);
if (key == NULL)
{
return;
}
/* If Key Pressed is Enter, send the Command to NSH stdin */
if (key[0] == 0xef && key[1] == 0xa2 && key[2] == 0xa2)
{
/* Read the NSH Input */
const char *cmd;
DEBUGASSERT(g_input != NULL);
cmd = lv_textarea_get_text(g_input);
if (cmd == NULL || cmd[0] == 0)
{
return;
}
/* Send the Command to NSH stdin */
DEBUGASSERT(g_nsh_stdin[WRITE_PIPE] != 0);
ret = write(g_nsh_stdin[WRITE_PIPE], cmd, strlen(cmd));
DEBUGASSERT(ret == strlen(cmd));
/* Erase the NSH Input */
lv_textarea_set_text(g_input, "");
}
}
}
/****************************************************************************
* Name: has_input
*
* Description:
* Return true if the File Descriptor has data to be read.
*
* Input Parameters:
* fd - File Descriptor to be checked
*
* Returned Value:
* True if File Descriptor has data to be read; False otherwise
*
****************************************************************************/
static bool has_input(int fd)
{
int ret;
/* Poll the File Descriptor for input */
struct pollfd fdp;
fdp.fd = fd;
fdp.events = POLLIN;
ret = poll(&fdp, /* File Descriptors */
1, /* Number of File Descriptors */
0); /* Poll Timeout (Milliseconds) */
if (ret > 0)
{
/* If poll is OK and there is input */
if ((fdp.revents & POLLIN) != 0)
{
/* Report that there is input */
return true;
}
/* Else report no input */
return false;
}
else if (ret == 0)
{
/* If timeout, report no input */
return false;
}
else if (ret < 0)
{
/* Handle error */
_err("poll failed: %d, fd=%d\n", ret, fd);
return false;
}
/* Never comes here */
DEBUGPANIC();
return false;
}
/****************************************************************************
* Name: remove_escape_codes
*
* Description:
* Remove ANSI Escape Codes from the string. Assumes that buf[len] is 0.
*
* Input Parameters:
* buf - String to be processed
* len - Length of string
*
* Returned Value:
* None
*
****************************************************************************/
static void remove_escape_codes(char *buf, int len)
{
int i;
int j;
for (i = 0; i < len; i++)
{
/* Escape Code looks like 0x1b 0x5b 0x4b */
if (buf[i] == 0x1b)
{
/* Remove 3 bytes */
for (j = i; j + 2 < len; j++)
{
buf[j] = buf[j + 3];
}
}
}
}
/****************************************************************************
* Public Functions
****************************************************************************/
/****************************************************************************
* Name: main or lvglterm_main
*
* Description:
* Start an LVGL Terminal that runs interactive commands with NSH Shell.
* NSH Commands are entered through an LVGL Keyboard. NSH Output is
* rendered in an LVGL Widget.
*
* Input Parameters:
* Standard argc and argv
*
* Returned Value:
* Zero on success; a positive, non-zero value on failure.
*
****************************************************************************/
int main(int argc, FAR char *argv[])
{
int ret;
#ifdef NEED_BOARDINIT
/* Perform board-specific driver initialization */
boardctl(BOARDIOC_INIT, 0);
#ifdef CONFIG_BOARDCTL_FINALINIT
/* Perform architecture-specific final-initialization (if configured) */
boardctl(BOARDIOC_FINALINIT, 0);
#endif
#endif
/* LVGL initialization */
lv_init();
/* LVGL port initialization */
lv_port_init();
/* Create the LVGL Widgets */
ret = create_terminal();
if (ret < 0)
{
return EXIT_FAILURE;
}
/* Handle LVGL tasks */
while (1)
{
uint32_t idle;
idle = lv_timer_handler();
/* Minimum sleep of 1ms */
idle = idle ? idle : 1;
usleep(idle * 1000);
}
return EXIT_SUCCESS;
}