[components/driver] update serial (#8567)
This commit is contained in:
parent
7eccdb471e
commit
00c6800e4e
|
@ -12,6 +12,7 @@
|
|||
#include <rtdevice.h>
|
||||
#include <drivers/platform.h>
|
||||
#include <drivers/core/bus.h>
|
||||
#include "../serial/serial_dm.h"
|
||||
|
||||
#define DBG_TAG "rtdm.ofw"
|
||||
#define DBG_LVL DBG_INFO
|
||||
|
@ -61,7 +62,7 @@ struct rt_ofw_stub *rt_ofw_stub_probe_range(struct rt_ofw_node *np,
|
|||
|
||||
static const char *ofw_console_serial_find(char *dst_con, struct rt_ofw_node *np)
|
||||
{
|
||||
rt_object_t rt_obj;
|
||||
rt_object_t rt_obj = RT_NULL;
|
||||
const char *ofw_name = RT_NULL;
|
||||
struct rt_serial_device *rt_serial = rt_ofw_data(np);
|
||||
|
||||
|
@ -168,7 +169,7 @@ static const char *ofw_console_tty_find(char *dst_con, const char *con)
|
|||
rt_err_t rt_ofw_console_setup(void)
|
||||
{
|
||||
rt_err_t err = -RT_ENOSYS;
|
||||
char con_name[RT_NAME_MAX];
|
||||
char con_name[RT_NAME_MAX], *options = RT_NULL;
|
||||
const char *ofw_name = RT_NULL, *stdout_path, *con;
|
||||
|
||||
/* chosen.console > chosen.stdout-path > RT_CONSOLE_DEVICE_NAME */
|
||||
|
@ -190,6 +191,18 @@ rt_err_t rt_ofw_console_setup(void)
|
|||
|
||||
if (ofw_name)
|
||||
{
|
||||
const char *ch = con;
|
||||
|
||||
while (*ch && *ch != ' ')
|
||||
{
|
||||
if (*ch++ == ',')
|
||||
{
|
||||
options = (char *)ch;
|
||||
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
err = RT_EOK;
|
||||
break;
|
||||
}
|
||||
|
@ -230,6 +243,18 @@ rt_err_t rt_ofw_console_setup(void)
|
|||
|
||||
rt_console_set_device(con);
|
||||
|
||||
if (options)
|
||||
{
|
||||
rt_device_t con_dev = rt_console_get_device();
|
||||
|
||||
if (con_dev)
|
||||
{
|
||||
struct serial_configure con_conf = serial_cfg_from_args(options);
|
||||
|
||||
rt_device_control(con_dev, RT_DEVICE_CTRL_CONFIG, &con_conf);
|
||||
}
|
||||
}
|
||||
|
||||
rt_fdt_earlycon_kick(FDT_EARLYCON_KICK_COMPLETED);
|
||||
|
||||
LOG_I("Console: %s (%s)", con, ofw_name ? ofw_name : "<unknown>");
|
||||
|
|
|
@ -3,12 +3,19 @@ from building import *
|
|||
cwd = GetCurrentDir()
|
||||
CPPPATH = [cwd + '/../include']
|
||||
group = []
|
||||
if GetDepend(['RT_USING_SERIAL']):
|
||||
if GetDepend(['RT_USING_SERIAL_V2']):
|
||||
src = Glob('serial_v2.c')
|
||||
group = DefineGroup('DeviceDrivers', src, depend = ['RT_USING_SERIAL_V2'], CPPPATH = CPPPATH)
|
||||
else:
|
||||
src = Glob('serial.c')
|
||||
group = DefineGroup('DeviceDrivers', src, depend = ['RT_USING_SERIAL'], CPPPATH = CPPPATH)
|
||||
src = []
|
||||
|
||||
if not GetDepend(['RT_USING_SERIAL']):
|
||||
Return('group')
|
||||
|
||||
if GetDepend(['RT_USING_SERIAL_V2']):
|
||||
src += ['serial_v2.c']
|
||||
else:
|
||||
src += ['serial.c']
|
||||
|
||||
if GetDepend(['RT_USING_DM']):
|
||||
src += ['serial_dm.c']
|
||||
|
||||
group = DefineGroup('DeviceDrivers', src, depend = [''], CPPPATH = CPPPATH)
|
||||
|
||||
Return('group')
|
||||
|
|
|
@ -0,0 +1,154 @@
|
|||
/*
|
||||
* Copyright (c) 2006-2022, RT-Thread Development Team
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*
|
||||
* Change Logs:
|
||||
* Date Author Notes
|
||||
* 2022-11-16 GuEe-GUI first version
|
||||
*/
|
||||
|
||||
#include <rtatomic.h>
|
||||
#include "serial_dm.h"
|
||||
|
||||
int serial_dev_set_name(struct rt_serial_device *sdev)
|
||||
{
|
||||
int id = -1;
|
||||
static int uid_min = -1;
|
||||
static volatile rt_atomic_t uid = 0;
|
||||
|
||||
RT_ASSERT(sdev != RT_NULL);
|
||||
|
||||
#ifdef RT_USING_OFW
|
||||
if (sdev->parent.ofw_node)
|
||||
{
|
||||
id = rt_ofw_get_alias_id(sdev->parent.ofw_node, "serial");
|
||||
|
||||
if (id < 0)
|
||||
{
|
||||
id = rt_ofw_get_alias_id(sdev->parent.ofw_node, "uart");
|
||||
}
|
||||
|
||||
if (uid_min < 0)
|
||||
{
|
||||
uid_min = rt_ofw_get_alias_last_id("serial");
|
||||
|
||||
if (uid_min < 0)
|
||||
{
|
||||
uid_min = rt_ofw_get_alias_last_id("uart");
|
||||
}
|
||||
|
||||
uid_min = uid_min < 0 ? 0 : (uid_min + 1);
|
||||
|
||||
rt_hw_atomic_store(&uid, uid_min);
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
||||
if (id < 0)
|
||||
{
|
||||
id = (int)rt_hw_atomic_add(&uid, 1);
|
||||
}
|
||||
|
||||
return rt_dm_dev_set_name(&sdev->parent, "uart%u", id);
|
||||
}
|
||||
|
||||
void *serial_base_from_args(char *str)
|
||||
{
|
||||
rt_ubase_t base = 0;
|
||||
|
||||
while (*str && !(*str == 'x' || *str == 'X'))
|
||||
{
|
||||
++str;
|
||||
}
|
||||
++str;
|
||||
|
||||
/* The str may get from bootargs that we need check it */
|
||||
while (*str)
|
||||
{
|
||||
if ((*str >= 'a' && *str <= 'f') || (*str >= 'A' && *str <= 'F'))
|
||||
{
|
||||
base = (base << 4) | (((*str | ' ') - 'a') + 10);
|
||||
}
|
||||
else if (*str >= '0' && *str <= '9')
|
||||
{
|
||||
base = (base << 4) | (*str - '0');
|
||||
}
|
||||
else break;
|
||||
|
||||
++str;
|
||||
}
|
||||
|
||||
return (void *)base;
|
||||
}
|
||||
|
||||
struct serial_configure serial_cfg_from_args(char *str)
|
||||
{
|
||||
struct serial_configure cfg = RT_SERIAL_CONFIG_DEFAULT;
|
||||
|
||||
/* Format baudrate/parity/bits/flow (BBBBPNF), Default is 115200n8 */
|
||||
if (str && *str)
|
||||
{
|
||||
rt_uint32_t baudrate = 0;
|
||||
|
||||
/* BBBB is the speed */
|
||||
while (*str && (*str >= '0' && *str <= '9'))
|
||||
{
|
||||
baudrate *= 10;
|
||||
baudrate += *str - '0';
|
||||
++str;
|
||||
}
|
||||
|
||||
if (baudrate)
|
||||
{
|
||||
cfg.baud_rate = baudrate;
|
||||
}
|
||||
|
||||
/* P is parity (n/o/e) */
|
||||
switch (*str)
|
||||
{
|
||||
case 'n':
|
||||
cfg.parity = PARITY_NONE;
|
||||
break;
|
||||
case 'o':
|
||||
cfg.parity = PARITY_ODD;
|
||||
break;
|
||||
case 'e':
|
||||
cfg.parity = PARITY_EVEN;
|
||||
break;
|
||||
default:
|
||||
--str;
|
||||
break;
|
||||
}
|
||||
++str;
|
||||
|
||||
/* N is number of bits */
|
||||
if (*str && (*str >= '0' && *str <= '9'))
|
||||
{
|
||||
cfg.data_bits = *str - '0';
|
||||
++str;
|
||||
}
|
||||
|
||||
/* F is flow ontrol ('r' for RTS) */
|
||||
if (*str)
|
||||
{
|
||||
cfg.flowcontrol = (*str == 'r' ? RT_SERIAL_FLOWCONTROL_CTSRTS : RT_SERIAL_FLOWCONTROL_NONE);
|
||||
++str;
|
||||
}
|
||||
|
||||
#ifdef RT_USING_OFW
|
||||
if (*str == '\0')
|
||||
{
|
||||
const char earlycon_magic[] = { 'O', 'F', 'W', '\0' };
|
||||
|
||||
if (!rt_strcmp(++str, earlycon_magic))
|
||||
{
|
||||
/* Is OFW earlycon, we should ACK it */
|
||||
rt_memset(str, 0, RT_ARRAY_SIZE(earlycon_magic));
|
||||
}
|
||||
}
|
||||
#endif
|
||||
}
|
||||
|
||||
return cfg;
|
||||
}
|
|
@ -0,0 +1,29 @@
|
|||
/*
|
||||
* Copyright (c) 2006-2022, RT-Thread Development Team
|
||||
*
|
||||
* SPDX-License-Identifier: Apache-2.0
|
||||
*
|
||||
* Change Logs:
|
||||
* Date Author Notes
|
||||
* 2022-11-16 GuEe-GUI first version
|
||||
*/
|
||||
|
||||
#ifndef __SERIAL_DM_H__
|
||||
#define __SERIAL_DM_H__
|
||||
|
||||
#include <rtthread.h>
|
||||
#include <rtdevice.h>
|
||||
#include <string.h>
|
||||
|
||||
int serial_dev_set_name(struct rt_serial_device *sdev);
|
||||
|
||||
void *serial_base_from_args(char *str);
|
||||
struct serial_configure serial_cfg_from_args(char *str);
|
||||
|
||||
#define serial_for_each_args(arg, args) \
|
||||
for (char *context = (arg = (typeof(arg))args, (void *)RT_NULL), \
|
||||
*context_end = rt_strchrnul((char *)args, ' '); \
|
||||
(arg = strtok_r(arg, ",", &context)) && arg < context_end; \
|
||||
arg = RT_NULL)
|
||||
|
||||
#endif /* __SERIAL_DM_H__ */
|
|
@ -519,3 +519,8 @@ rt_weak void rt_hw_secondary_cpu_idle_exec(void)
|
|||
rt_hw_wfe();
|
||||
}
|
||||
#endif
|
||||
|
||||
void rt_hw_console_output(const char *str)
|
||||
{
|
||||
rt_fdt_earlycon_output(str);
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue