2012-03-30 20:42:40 +02:00
|
|
|
/****************************************************************************
|
2021-03-08 22:39:04 +01:00
|
|
|
* graphics/nxterm/nxterm_vt100.c
|
2012-03-30 20:42:40 +02:00
|
|
|
*
|
2024-09-10 09:30:58 +02:00
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*
|
2021-02-05 11:12:53 +01:00
|
|
|
* 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
|
2012-03-30 20:42:40 +02:00
|
|
|
*
|
2021-02-05 11:12:53 +01:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2012-03-30 20:42:40 +02:00
|
|
|
*
|
2021-02-05 11:12:53 +01:00
|
|
|
* 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.
|
2012-03-30 20:42:40 +02:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Included Files
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
#include <nuttx/config.h>
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
#include <nuttx/vt100.h>
|
|
|
|
|
2014-09-20 23:53:28 +02:00
|
|
|
#include "nxterm.h"
|
2012-03-30 20:42:40 +02:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Types
|
|
|
|
****************************************************************************/
|
|
|
|
|
2014-09-20 23:53:28 +02:00
|
|
|
typedef int (*seqhandler_t)(FAR struct nxterm_state_s *priv);
|
2012-03-30 20:42:40 +02:00
|
|
|
|
|
|
|
struct vt100_sequence_s
|
|
|
|
{
|
|
|
|
FAR const char *seq;
|
|
|
|
seqhandler_t handler;
|
|
|
|
uint8_t size;
|
|
|
|
};
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Function Prototypes
|
|
|
|
****************************************************************************/
|
|
|
|
|
2014-09-20 23:53:28 +02:00
|
|
|
static int nxterm_erasetoeol(FAR struct nxterm_state_s *priv);
|
2012-03-30 20:42:40 +02:00
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Data
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/* All recognized VT100 escape sequences. Very little as present, this is
|
|
|
|
* a placeholder for a future, more complete VT100 emulation.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* <esc>[K is the VT100 command erases to the end of the line. */
|
|
|
|
|
|
|
|
static const char g_erasetoeol[] = VT100_CLEAREOL;
|
|
|
|
|
|
|
|
/* The list of all VT100 sequences supported by the emulation */
|
|
|
|
|
|
|
|
static const struct vt100_sequence_s g_vt100sequences[] =
|
|
|
|
{
|
2014-09-20 23:53:28 +02:00
|
|
|
{g_erasetoeol, nxterm_erasetoeol, sizeof(g_erasetoeol)},
|
2012-03-30 20:42:40 +02:00
|
|
|
{NULL, NULL, 0}
|
|
|
|
};
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Private Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
2014-09-20 23:53:28 +02:00
|
|
|
* Name: nxterm_erasetoeol
|
2012-03-30 20:42:40 +02:00
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Handle the erase-to-eol VT100 escapte sequence
|
|
|
|
*
|
|
|
|
* Input Parameters:
|
|
|
|
* priv - Driver data structure
|
|
|
|
*
|
|
|
|
* Returned Value:
|
|
|
|
* The index of the match in g_vt100sequences[]
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2014-09-20 23:53:28 +02:00
|
|
|
static int nxterm_erasetoeol(FAR struct nxterm_state_s *priv)
|
2012-03-30 20:42:40 +02:00
|
|
|
{
|
|
|
|
/* Does nothing yet (other than consume the sequence) */
|
|
|
|
|
|
|
|
return OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2014-09-20 23:53:28 +02:00
|
|
|
* Name: nxterm_vt100part
|
2012-03-30 20:42:40 +02:00
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Return the next entry that is a partial match to the sequence.
|
|
|
|
*
|
|
|
|
* Input Parameters:
|
|
|
|
* priv - Driver data structure
|
|
|
|
* seqsize - The number of bytes in the sequence
|
|
|
|
* startndx - The index to start searching
|
|
|
|
*
|
|
|
|
* Returned Value:
|
|
|
|
* A pointer to the matching sequence in g_vt100sequences[]
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
FAR const struct vt100_sequence_s *
|
2014-09-20 23:53:28 +02:00
|
|
|
nxterm_vt100part(FAR struct nxterm_state_s *priv, int seqsize)
|
2012-03-30 20:42:40 +02:00
|
|
|
{
|
|
|
|
FAR const struct vt100_sequence_s *seq;
|
|
|
|
int ndx;
|
|
|
|
|
|
|
|
/* Search from the beginning of the sequence table */
|
|
|
|
|
|
|
|
for (ndx = 0; g_vt100sequences[ndx].seq; ndx++)
|
|
|
|
{
|
|
|
|
/* Is this sequence big enough? */
|
|
|
|
|
|
|
|
seq = &g_vt100sequences[ndx];
|
|
|
|
if (seq->size >= seqsize)
|
|
|
|
{
|
|
|
|
/* Yes... are the first 'seqsize' bytes the same */
|
|
|
|
|
|
|
|
if (memcmp(seq->seq, priv->seq, seqsize) == 0)
|
|
|
|
{
|
|
|
|
/* Yes.. return the match */
|
|
|
|
|
|
|
|
return seq;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-02-05 09:05:48 +01:00
|
|
|
|
2012-03-30 20:42:40 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
2014-09-20 23:53:28 +02:00
|
|
|
* Name: nxterm_vt100seq
|
2012-03-30 20:42:40 +02:00
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Determine if the new sequence is a part of a supported VT100 escape
|
|
|
|
* sequence.
|
|
|
|
*
|
|
|
|
* Input Parameters:
|
|
|
|
* priv - Driver data structure
|
|
|
|
* seqsize - The number of bytes in the sequence
|
|
|
|
*
|
|
|
|
* Returned Value:
|
2014-09-20 23:53:28 +02:00
|
|
|
* state - See enum nxterm_vt100state_e;
|
2012-03-30 20:42:40 +02:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2021-02-05 09:05:48 +01:00
|
|
|
static enum nxterm_vt100state_e nxterm_vt100seq(
|
|
|
|
FAR struct nxterm_state_s *priv,
|
|
|
|
int seqsize)
|
2012-03-30 20:42:40 +02:00
|
|
|
{
|
|
|
|
FAR const struct vt100_sequence_s *seq;
|
2014-09-20 23:53:28 +02:00
|
|
|
enum nxterm_vt100state_e ret;
|
2012-03-30 20:42:40 +02:00
|
|
|
|
|
|
|
/* Is there any VT100 escape sequence that matches what we have
|
|
|
|
* buffered so far?
|
|
|
|
*/
|
|
|
|
|
2014-09-20 23:53:28 +02:00
|
|
|
seq = nxterm_vt100part(priv, seqsize);
|
2012-03-30 20:42:40 +02:00
|
|
|
if (seq)
|
|
|
|
{
|
|
|
|
/* Yes.. if the size of that escape sequence is the same as what we
|
|
|
|
* have buffered, then we have an exact match.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (seq->size == seqsize)
|
|
|
|
{
|
|
|
|
/* Process the VT100 sequence */
|
|
|
|
|
|
|
|
seq->handler(priv);
|
|
|
|
priv->nseq = 0;
|
|
|
|
return VT100_PROCESSED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* The 'seqsize' is still smaller than the potential match(es). We
|
|
|
|
* will need to collect more characters before we can make a decision.
|
2020-02-23 09:50:23 +01:00
|
|
|
* Return an indication that we have consumed the character.
|
2012-03-30 20:42:40 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
return VT100_CONSUMED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* We get here on a failure. The buffer sequence is not part of any
|
|
|
|
* supported VT100 escape sequence. If seqsize > 1 then we need to
|
|
|
|
* return a special value because we have to re-process the buffered
|
|
|
|
* data.
|
|
|
|
*/
|
|
|
|
|
|
|
|
ret = seqsize > 1 ? VT100_ABORT : VT100_NOT_CONSUMED;
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
* Public Functions
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
/****************************************************************************
|
2014-09-20 23:53:28 +02:00
|
|
|
* Name: nxterm_vt100
|
2012-03-30 20:42:40 +02:00
|
|
|
*
|
|
|
|
* Description:
|
|
|
|
* Test if the newly received byte is part of a VT100 escape sequence
|
|
|
|
*
|
|
|
|
* Input Parameters:
|
|
|
|
* priv - Driver data structure
|
|
|
|
* ch - The newly received character
|
|
|
|
*
|
|
|
|
* Returned Value:
|
2014-09-20 23:53:28 +02:00
|
|
|
* state - See enum nxterm_vt100state_e;
|
2012-03-30 20:42:40 +02:00
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2021-02-05 09:05:48 +01:00
|
|
|
enum nxterm_vt100state_e nxterm_vt100(FAR struct nxterm_state_s *priv,
|
|
|
|
char ch)
|
2012-03-30 20:42:40 +02:00
|
|
|
{
|
2014-09-20 23:53:28 +02:00
|
|
|
enum nxterm_vt100state_e ret;
|
2012-03-30 20:42:40 +02:00
|
|
|
int seqsize;
|
|
|
|
|
|
|
|
DEBUGASSERT(priv && priv->nseq < VT100_MAX_SEQUENCE);
|
|
|
|
|
|
|
|
/* If we have no buffered characters, then 'ch' must be the first character
|
|
|
|
* of an escape sequence.
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (priv->nseq < 1)
|
|
|
|
{
|
|
|
|
/* The first character of an escape sequence must be an an escape
|
|
|
|
* character (duh).
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (ch != ASCII_ESC)
|
|
|
|
{
|
|
|
|
return VT100_NOT_CONSUMED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Add the escape character to the buffer but don't bother with any
|
|
|
|
* further checking.
|
|
|
|
*/
|
|
|
|
|
|
|
|
priv->seq[0] = ASCII_ESC;
|
|
|
|
priv->nseq = 1;
|
|
|
|
return VT100_CONSUMED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Temporarily add the next character to the buffer */
|
|
|
|
|
|
|
|
seqsize = priv->nseq;
|
|
|
|
priv->seq[seqsize] = ch;
|
|
|
|
|
|
|
|
/* Then check if this sequence is part of an a valid escape sequence */
|
|
|
|
|
|
|
|
seqsize++;
|
2014-09-20 23:53:28 +02:00
|
|
|
ret = nxterm_vt100seq(priv, seqsize);
|
2012-03-30 20:42:40 +02:00
|
|
|
if (ret == VT100_CONSUMED)
|
|
|
|
{
|
|
|
|
/* The newly added character is indeed part of a VT100 escape sequence
|
|
|
|
* (which is still incomplete). Keep it in the buffer.
|
|
|
|
*/
|
|
|
|
|
|
|
|
priv->nseq = seqsize;
|
|
|
|
}
|
2021-02-05 09:05:48 +01:00
|
|
|
|
2012-03-30 20:42:40 +02:00
|
|
|
return ret;
|
|
|
|
}
|