添加rtthread相关代码
This commit is contained in:
		
							
								
								
									
										77
									
								
								riscv/rtthread/components/net/at/Kconfig
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										77
									
								
								riscv/rtthread/components/net/at/Kconfig
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,77 @@ | ||||
| menuconfig RT_USING_AT | ||||
|     bool "Enable AT commands" | ||||
|     default n | ||||
|  | ||||
| if RT_USING_AT | ||||
|  | ||||
|     config AT_DEBUG | ||||
|         bool "Enable debug log output" | ||||
|         default n | ||||
|  | ||||
|     config AT_USING_SERVER | ||||
|         bool "Enable AT commands server" | ||||
|         default n | ||||
|  | ||||
|     if AT_USING_SERVER | ||||
|  | ||||
|         config AT_SERVER_DEVICE | ||||
|             string "Server device name" | ||||
|             default "uart3" | ||||
|  | ||||
|         config AT_SERVER_RECV_BUFF_LEN | ||||
|             int "The maximum length of server data accepted" | ||||
|             default 256 | ||||
|  | ||||
|         config AT_SERVER_SEND_BUFF_LEN | ||||
|             int "The maximum length of server commands buffer" | ||||
|             default 256 | ||||
|  | ||||
|     endif | ||||
|  | ||||
|     config AT_USING_CLIENT | ||||
|         bool "Enable AT commands client" | ||||
|         default n | ||||
|  | ||||
|     if AT_USING_CLIENT | ||||
|  | ||||
|         config AT_CLIENT_NUM_MAX | ||||
|             int "The maximum number of supported clients" | ||||
|             default 1 | ||||
|             range 1 65535 | ||||
|  | ||||
|         config AT_USING_SOCKET | ||||
|             bool "Enable BSD Socket API support by AT commnads" | ||||
|             select RT_USING_SAL | ||||
|             select SAL_USING_AT | ||||
|             default n | ||||
|  | ||||
|         if AT_USING_SOCKET | ||||
|  | ||||
|             config AT_USING_SOCKET_SERVER | ||||
|                 bool "Enable BSD Socket API support about AT server" | ||||
|                 default n | ||||
|  | ||||
|         endif | ||||
|  | ||||
|     endif | ||||
|  | ||||
|     if AT_USING_SERVER || AT_USING_CLIENT | ||||
|  | ||||
|         config AT_USING_CLI | ||||
|             bool "Enable CLI(Command-Line Interface) for AT commands" | ||||
|             default y | ||||
|             depends on RT_USING_FINSH | ||||
|  | ||||
|        config AT_PRINT_RAW_CMD | ||||
|             bool "Enable print RAW format AT command communication data" | ||||
|             default n | ||||
|  | ||||
|     endif | ||||
|  | ||||
|     config AT_SW_VERSION_NUM | ||||
|         hex | ||||
|         default 0x10301 | ||||
|         help | ||||
|             software module version number | ||||
|  | ||||
| endif | ||||
							
								
								
									
										26
									
								
								riscv/rtthread/components/net/at/SConscript
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										26
									
								
								riscv/rtthread/components/net/at/SConscript
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,26 @@ | ||||
| from building import * | ||||
|  | ||||
| cwd = GetCurrentDir() | ||||
| path = [cwd + '/include'] | ||||
|  | ||||
| src = Glob('src/at_utils.c') | ||||
|  | ||||
| if GetDepend(['AT_USING_CLI']): | ||||
|     src += Glob('src/at_cli.c') | ||||
|  | ||||
| if GetDepend(['AT_USING_SERVER']): | ||||
|     src += Split(''' | ||||
|     src/at_server.c | ||||
|     src/at_base_cmd.c | ||||
|     ''') | ||||
|  | ||||
| if GetDepend(['AT_USING_CLIENT']): | ||||
|     src += Glob('src/at_client.c') | ||||
|  | ||||
| if GetDepend(['AT_USING_SOCKET']): | ||||
|     src += Glob('at_socket/*.c') | ||||
|     path += [cwd + '/at_socket'] | ||||
|  | ||||
| group = DefineGroup('AT', src, depend = ['RT_USING_AT'], CPPPATH = path) | ||||
|  | ||||
| Return('group') | ||||
							
								
								
									
										1533
									
								
								riscv/rtthread/components/net/at/at_socket/at_socket.c
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										1533
									
								
								riscv/rtthread/components/net/at/at_socket/at_socket.c
									
									
									
									
									
										Executable file
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										200
									
								
								riscv/rtthread/components/net/at/at_socket/at_socket.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										200
									
								
								riscv/rtthread/components/net/at/at_socket/at_socket.h
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,200 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2018-06-06     chenYong     first version | ||||
|  * 2022-06-02     xianxistu    add implement about "AT server" | ||||
|  */ | ||||
|  | ||||
| #ifndef __AT_SOCKET_H__ | ||||
| #define __AT_SOCKET_H__ | ||||
|  | ||||
| #include <rtthread.h> | ||||
| #include <rtdevice.h> | ||||
| #include <rthw.h> | ||||
|  | ||||
| #include <netdb.h> | ||||
| #include <sys/socket.h> | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| extern "C" { | ||||
| #endif | ||||
|  | ||||
| #ifndef AT_SOCKET_RECV_BFSZ | ||||
| #define AT_SOCKET_RECV_BFSZ            512 | ||||
| #endif | ||||
|  | ||||
| #define AT_DEFAULT_RECVMBOX_SIZE       10 | ||||
| #define AT_DEFAULT_ACCEPTMBOX_SIZE     10 | ||||
|  | ||||
| /* sal socket magic word */ | ||||
| #define AT_SOCKET_MAGIC                0xA100 | ||||
|  | ||||
| #ifdef AT_USING_SOCKET_SERVER | ||||
| #define AT_SOCKET_INFO_LEN (sizeof("SOCKET:") + 4) | ||||
| #endif | ||||
|  | ||||
| /* Current state of the AT socket. */ | ||||
| enum at_socket_state | ||||
| { | ||||
|     AT_SOCKET_NONE, | ||||
|     AT_SOCKET_OPEN, | ||||
|     AT_SOCKET_LISTEN, | ||||
|     AT_SOCKET_CONNECT, | ||||
|     AT_SOCKET_CLOSED | ||||
| }; | ||||
|  | ||||
| enum at_socket_type | ||||
| { | ||||
|     AT_SOCKET_INVALID   = 0, | ||||
|     AT_SOCKET_TCP       = 0x10,      /* TCP IPv4 */ | ||||
|     AT_SOCKET_UDP       = 0x20,      /* UDP IPv4 */ | ||||
| }; | ||||
|  | ||||
| typedef enum | ||||
| { | ||||
|     AT_SOCKET_EVT_RECV, | ||||
|     AT_SOCKET_EVT_CLOSED, | ||||
| #ifdef AT_USING_SOCKET_SERVER | ||||
|     AT_SOCKET_EVT_CONNECTED, | ||||
| #endif | ||||
| } at_socket_evt_t; | ||||
|  | ||||
| struct at_socket; | ||||
| struct at_device; | ||||
|  | ||||
| typedef void (*at_evt_cb_t)(struct at_socket *socket, at_socket_evt_t event, const char *buff, size_t bfsz); | ||||
|  | ||||
| /* A callback prototype to inform about events for AT socket */ | ||||
| typedef void (* at_socket_callback)(struct at_socket *conn, int event, uint16_t len); | ||||
|  | ||||
| /* AT socket operations function */ | ||||
| struct at_socket_ops | ||||
| { | ||||
|     int (*at_connect)(struct at_socket *socket, char *ip, int32_t port, enum at_socket_type type, rt_bool_t is_client); | ||||
|     int (*at_closesocket)(struct at_socket *socket); | ||||
|     int (*at_send)(struct at_socket *socket, const char *buff, size_t bfsz, enum at_socket_type type); | ||||
|     int (*at_domain_resolve)(const char *name, char ip[16]); | ||||
|     void (*at_set_event_cb)(at_socket_evt_t event, at_evt_cb_t cb); | ||||
|     int (*at_socket)(struct at_device *device, enum at_socket_type type); | ||||
| #ifdef AT_USING_SOCKET_SERVER | ||||
|     int (*at_listen)(struct at_socket *socket, int backlog); | ||||
| #endif | ||||
| }; | ||||
|  | ||||
| /* AT receive package list structure */ | ||||
| struct at_recv_pkt | ||||
| { | ||||
|     rt_slist_t list; | ||||
|     size_t bfsz_totle; | ||||
|     size_t bfsz_index; | ||||
|     char *buff; | ||||
| }; | ||||
| typedef struct at_recv_pkt *at_recv_pkt_t; | ||||
|  | ||||
| #ifdef AT_USING_SOCKET_SERVER | ||||
| struct at_listen_state | ||||
| { | ||||
|     uint16_t is_listen; | ||||
|     uint16_t port; | ||||
| }; | ||||
| #endif | ||||
|  | ||||
| struct at_socket | ||||
| { | ||||
|     /* AT socket magic word */ | ||||
|     uint32_t magic; | ||||
|  | ||||
|     int socket; | ||||
| #ifdef AT_USING_SOCKET_SERVER | ||||
|     struct at_listen_state listen; | ||||
| #endif | ||||
|  | ||||
|     /* device releated information for the socket */ | ||||
|     void *device; | ||||
|     /* type of the AT socket (TCP, UDP or RAW) */ | ||||
|     enum at_socket_type type; | ||||
|     /* current state of the AT socket */ | ||||
|     enum at_socket_state state; | ||||
|     /* sockets operations */ | ||||
|     const struct at_socket_ops *ops; | ||||
|     /* receive semaphore, received data release semaphore */ | ||||
|     rt_sem_t recv_notice; | ||||
|     rt_mutex_t recv_lock; | ||||
|     rt_slist_t recvpkt_list; | ||||
|  | ||||
|     /* timeout to wait for send or received data in milliseconds */ | ||||
|     int32_t recv_timeout; | ||||
|     int32_t send_timeout; | ||||
|     /* A callback function that is informed about events for this AT socket */ | ||||
|     at_socket_callback callback; | ||||
|  | ||||
|     /* number of times data was received, set by event_callback() */ | ||||
|     uint16_t rcvevent; | ||||
|     /* number of times data was ACKed (free send buffer), set by event_callback() */ | ||||
|     uint16_t sendevent; | ||||
|     /* error happened for this socket, set by event_callback() */ | ||||
|     uint16_t errevent; | ||||
|  | ||||
| #ifdef SAL_USING_POSIX | ||||
|     rt_wqueue_t wait_head; | ||||
| #endif | ||||
|     rt_slist_t list; | ||||
|  | ||||
|     /* user-specific data */ | ||||
|     void *user_data; | ||||
| }; | ||||
|  | ||||
| int at_socket(int domain, int type, int protocol); | ||||
| int at_closesocket(int socket); | ||||
| int at_shutdown(int socket, int how); | ||||
| int at_bind(int socket, const struct sockaddr *name, socklen_t namelen); | ||||
| #ifdef AT_USING_SOCKET_SERVER | ||||
| int at_listen(int socket, int backlog); | ||||
| #endif | ||||
| int at_connect(int socket, const struct sockaddr *name, socklen_t namelen); | ||||
| #ifdef AT_USING_SOCKET_SERVER | ||||
| int at_accept(int socket, struct sockaddr *name, socklen_t *namelen); | ||||
| #endif | ||||
| int at_sendto(int socket, const void *data, size_t size, int flags, const struct sockaddr *to, socklen_t tolen); | ||||
| int at_send(int socket, const void *data, size_t size, int flags); | ||||
| int at_recvfrom(int socket, void *mem, size_t len, int flags, struct sockaddr *from, socklen_t *fromlen); | ||||
| int at_recv(int socket, void *mem, size_t len, int flags); | ||||
| int at_getsockopt(int socket, int level, int optname, void *optval, socklen_t *optlen); | ||||
| int at_setsockopt(int socket, int level, int optname, const void *optval, socklen_t optlen); | ||||
| struct hostent *at_gethostbyname(const char *name); | ||||
| int at_getaddrinfo(const char *nodename, const char *servname, const struct addrinfo *hints, struct addrinfo **res); | ||||
| void at_freeaddrinfo(struct addrinfo *ai); | ||||
|  | ||||
| struct at_socket *at_get_socket(int socket); | ||||
| #ifdef AT_USING_SOCKET_SERVER | ||||
| struct at_socket *at_get_base_socket(int base_socket); | ||||
| #endif | ||||
|  | ||||
| #ifndef RT_USING_SAL | ||||
|  | ||||
| #define socket(domain, type, protocol)                      at_socket(domain, type, protocol) | ||||
| #define closesocket(socket)                                 at_closesocket(socket) | ||||
| #define shutdown(socket, how)                               at_shutdown(socket, how) | ||||
| #define bind(socket, name, namelen)                         at_bind(socket, name, namelen) | ||||
| #define connect(socket, name, namelen)                      at_connect(socket, name, namelen) | ||||
| #define sendto(socket, data, size, flags, to, tolen)        at_sendto(socket, data, size, flags, to, tolen) | ||||
| #define send(socket, data, size, flags)                     at_send(socket, data, size, flags) | ||||
| #define recvfrom(socket, mem, len, flags, from, fromlen)    at_recvfrom(socket, mem, len, flags, from, fromlen) | ||||
| #define getsockopt(socket, level, optname, optval, optlen)  at_getsockopt(socket, level, optname, optval, optlen) | ||||
| #define setsockopt(socket, level, optname, optval, optlen)  at_setsockopt(socket, level, optname, optval, optlen) | ||||
|  | ||||
| #define gethostbyname(name)                                 at_gethostbyname(name) | ||||
| #define getaddrinfo(nodename, servname, hints, res)         at_getaddrinfo(nodename, servname, hints, res) | ||||
| #define freeaddrinfo(ai)                                    at_freeaddrinfo(ai) | ||||
|  | ||||
| #endif /* RT_USING_SAL */ | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #endif /* AT_SOCKET_H__ */ | ||||
							
								
								
									
										266
									
								
								riscv/rtthread/components/net/at/include/at.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										266
									
								
								riscv/rtthread/components/net/at/include/at.h
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,266 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2018-03-30     chenyong     first version | ||||
|  * 2018-08-17     chenyong     multiple client support | ||||
|  */ | ||||
|  | ||||
| #ifndef __AT_H__ | ||||
| #define __AT_H__ | ||||
|  | ||||
| #include <stddef.h> | ||||
| #include <rtthread.h> | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| extern "C" { | ||||
| #endif | ||||
|  | ||||
| #define AT_SW_VERSION                  "1.3.1" | ||||
|  | ||||
| #define AT_CMD_NAME_LEN                16 | ||||
|  | ||||
| #ifndef AT_SERVER_RECV_BUFF_LEN | ||||
| #define AT_SERVER_RECV_BUFF_LEN        256 | ||||
| #endif | ||||
|  | ||||
| #ifndef AT_SERVER_DEVICE | ||||
| #define AT_SERVER_DEVICE               "uart2" | ||||
| #endif | ||||
|  | ||||
| /* the maximum number of supported AT clients */ | ||||
| #ifndef AT_CLIENT_NUM_MAX | ||||
| #define AT_CLIENT_NUM_MAX              1 | ||||
| #endif | ||||
|  | ||||
| #define AT_CMD_EXPORT(_name_, _args_expr_, _test_, _query_, _setup_, _exec_)   \ | ||||
|     rt_used static const struct at_cmd __at_cmd_##_test_##_query_##_setup_##_exec_ rt_section("RtAtCmdTab") = \ | ||||
|     {                                                                          \ | ||||
|         _name_,                                                                \ | ||||
|         _args_expr_,                                                           \ | ||||
|         _test_,                                                                \ | ||||
|         _query_,                                                               \ | ||||
|         _setup_,                                                               \ | ||||
|         _exec_,                                                                \ | ||||
|     }; | ||||
|  | ||||
| enum at_status | ||||
| { | ||||
|     AT_STATUS_UNINITIALIZED = 0, | ||||
|     AT_STATUS_INITIALIZED, | ||||
|     AT_STATUS_CLI, | ||||
| }; | ||||
| typedef enum at_status at_status_t; | ||||
|  | ||||
| #ifdef AT_USING_SERVER | ||||
| enum at_result | ||||
| { | ||||
|     AT_RESULT_OK = 0,                  /* AT result is no error */ | ||||
|     AT_RESULT_FAILE = -1,              /* AT result have a generic error */ | ||||
|     AT_RESULT_NULL = -2,               /* AT result not need return */ | ||||
|     AT_RESULT_CMD_ERR = -3,            /* AT command format error or No way to execute */ | ||||
|     AT_RESULT_CHECK_FAILE = -4,        /* AT command expression format is error */ | ||||
|     AT_RESULT_PARSE_FAILE = -5,        /* AT command arguments parse is error */ | ||||
| }; | ||||
| typedef enum at_result at_result_t; | ||||
|  | ||||
| struct at_cmd | ||||
| { | ||||
|     char name[AT_CMD_NAME_LEN]; | ||||
|     char *args_expr; | ||||
|     at_result_t (*test)(void); | ||||
|     at_result_t (*query)(void); | ||||
|     at_result_t (*setup)(const char *args); | ||||
|     at_result_t (*exec)(void); | ||||
| }; | ||||
| typedef struct at_cmd *at_cmd_t; | ||||
|  | ||||
| struct at_server | ||||
| { | ||||
|     rt_device_t device; | ||||
|  | ||||
|     at_status_t status; | ||||
|     rt_err_t (*get_char)(struct at_server *server, char *ch, rt_int32_t timeout); | ||||
|     rt_bool_t echo_mode; | ||||
|  | ||||
|     char send_buffer[AT_SERVER_SEND_BUFF_LEN]; | ||||
|     char recv_buffer[AT_SERVER_RECV_BUFF_LEN]; | ||||
|     rt_size_t cur_recv_len; | ||||
|     rt_sem_t rx_notice; | ||||
|  | ||||
|     rt_thread_t parser; | ||||
|     void (*parser_entry)(struct at_server *server); | ||||
| }; | ||||
| typedef struct at_server *at_server_t; | ||||
| #endif /* AT_USING_SERVER */ | ||||
|  | ||||
| #ifdef AT_USING_CLIENT | ||||
| enum at_resp_status | ||||
| { | ||||
|     AT_RESP_OK = 0,                   /* AT response end is OK */ | ||||
|     AT_RESP_ERROR = -1,               /* AT response end is ERROR */ | ||||
|     AT_RESP_TIMEOUT = -2,             /* AT response is timeout */ | ||||
|     AT_RESP_BUFF_FULL= -3,            /* AT response buffer is full */ | ||||
| }; | ||||
| typedef enum at_resp_status at_resp_status_t; | ||||
|  | ||||
| struct at_response | ||||
| { | ||||
|     /* response buffer */ | ||||
|     char *buf; | ||||
|     /* the maximum response buffer size, it set by `at_create_resp()` function */ | ||||
|     rt_size_t buf_size; | ||||
|     /* the length of current response buffer */ | ||||
|     rt_size_t buf_len; | ||||
|     /* the number of setting response lines, it set by `at_create_resp()` function | ||||
|      * == 0: the response data will auto return when received 'OK' or 'ERROR' | ||||
|      * != 0: the response data will return when received setting lines number data */ | ||||
|     rt_size_t line_num; | ||||
|     /* the count of received response lines */ | ||||
|     rt_size_t line_counts; | ||||
|     /* the maximum response time */ | ||||
|     rt_int32_t timeout; | ||||
| }; | ||||
|  | ||||
| typedef struct at_response *at_response_t; | ||||
|  | ||||
| struct at_client; | ||||
|  | ||||
| /* URC(Unsolicited Result Code) object, such as: 'RING', 'READY' request by AT server */ | ||||
| struct at_urc | ||||
| { | ||||
|     const char *cmd_prefix; | ||||
|     const char *cmd_suffix; | ||||
|     void (*func)(struct at_client *client, const char *data, rt_size_t size); | ||||
| }; | ||||
| typedef struct at_urc *at_urc_t; | ||||
|  | ||||
| struct at_urc_table | ||||
| { | ||||
|     size_t urc_size; | ||||
|     const struct at_urc *urc; | ||||
| }; | ||||
| typedef struct at_urc *at_urc_table_t; | ||||
|  | ||||
| struct at_client | ||||
| { | ||||
|     rt_device_t device; | ||||
|  | ||||
|     at_status_t status; | ||||
|     char end_sign; | ||||
|  | ||||
|     char *send_buf; | ||||
|     /* The maximum supported send cmd length */ | ||||
|     rt_size_t send_bufsz; | ||||
|     /* The length of last cmd */ | ||||
|     rt_size_t last_cmd_len; | ||||
|  | ||||
|     /* the current received one line data buffer */ | ||||
|     char *recv_line_buf; | ||||
|     /* The length of the currently received one line data */ | ||||
|     rt_size_t recv_line_len; | ||||
|     /* The maximum supported receive data length */ | ||||
|     rt_size_t recv_bufsz; | ||||
|     rt_sem_t rx_notice; | ||||
|     rt_mutex_t lock; | ||||
|  | ||||
|     at_response_t resp; | ||||
|     rt_sem_t resp_notice; | ||||
|     at_resp_status_t resp_status; | ||||
|  | ||||
|     struct at_urc_table *urc_table; | ||||
|     rt_size_t urc_table_size; | ||||
|     const struct at_urc *urc; | ||||
|  | ||||
|     rt_thread_t parser; | ||||
| }; | ||||
| typedef struct at_client *at_client_t; | ||||
| #endif /* AT_USING_CLIENT */ | ||||
|  | ||||
| #ifdef AT_USING_SERVER | ||||
| /* AT server initialize and start */ | ||||
| int at_server_init(void); | ||||
|  | ||||
| /* AT server send command execute result to AT device */ | ||||
| void at_server_printf(const char *format, ...); | ||||
| void at_server_printfln(const char *format, ...); | ||||
| void at_server_print_result(at_result_t result); | ||||
| rt_size_t at_server_send(at_server_t server, const char *buf, rt_size_t size); | ||||
| rt_size_t at_server_recv(at_server_t server, char *buf, rt_size_t size, rt_int32_t timeout); | ||||
|  | ||||
| /* AT server request arguments parse */ | ||||
| int at_req_parse_args(const char *req_args, const char *req_expr, ...); | ||||
| #endif /* AT_USING_SERVER */ | ||||
|  | ||||
| #ifdef AT_USING_CLIENT | ||||
|  | ||||
| /* AT client initialize and start*/ | ||||
| int at_client_init(const char *dev_name, rt_size_t recv_bufsz, rt_size_t send_bufsz); | ||||
|  | ||||
| /* ========================== multiple AT client function ============================ */ | ||||
|  | ||||
| /* get AT client object */ | ||||
| at_client_t at_client_get(const char *dev_name); | ||||
| at_client_t at_client_get_first(void); | ||||
|  | ||||
| /* AT client wait for connection to external devices. */ | ||||
| int at_client_obj_wait_connect(at_client_t client, rt_uint32_t timeout); | ||||
|  | ||||
| /* AT client send or receive data */ | ||||
| rt_size_t at_client_obj_send(at_client_t client, const char *buf, rt_size_t size); | ||||
| rt_size_t at_client_obj_recv(at_client_t client, char *buf, rt_size_t size, rt_int32_t timeout); | ||||
|  | ||||
| /* set AT client a line end sign */ | ||||
| void at_obj_set_end_sign(at_client_t client, char ch); | ||||
|  | ||||
| /* Set URC(Unsolicited Result Code) table */ | ||||
| int at_obj_set_urc_table(at_client_t client, const struct at_urc * table, rt_size_t size); | ||||
|  | ||||
| /* AT client send commands to AT server and waiter response */ | ||||
| int at_obj_exec_cmd(at_client_t client, at_response_t resp, const char *cmd_expr, ...); | ||||
|  | ||||
| /* AT response object create and delete */ | ||||
| at_response_t at_create_resp(rt_size_t buf_size, rt_size_t line_num, rt_int32_t timeout); | ||||
| void at_delete_resp(at_response_t resp); | ||||
| at_response_t at_resp_set_info(at_response_t resp, rt_size_t buf_size, rt_size_t line_num, rt_int32_t timeout); | ||||
|  | ||||
| /* AT response line buffer get and parse response buffer arguments */ | ||||
| const char *at_resp_get_line(at_response_t resp, rt_size_t resp_line); | ||||
| const char *at_resp_get_line_by_kw(at_response_t resp, const char *keyword); | ||||
| int at_resp_parse_line_args(at_response_t resp, rt_size_t resp_line, const char *resp_expr, ...); | ||||
| int at_resp_parse_line_args_by_kw(at_response_t resp, const char *keyword, const char *resp_expr, ...); | ||||
|  | ||||
| /* ========================== single AT client function ============================ */ | ||||
|  | ||||
| /** | ||||
|  * NOTE: These functions can be used directly when there is only one AT client. | ||||
|  * If there are multiple AT Client in the program, these functions can operate on the first initialized AT client. | ||||
|  */ | ||||
|  | ||||
| #define at_exec_cmd(resp, ...)                   at_obj_exec_cmd(at_client_get_first(), resp, __VA_ARGS__) | ||||
| #define at_client_wait_connect(timeout)          at_client_obj_wait_connect(at_client_get_first(), timeout) | ||||
| #define at_client_send(buf, size)                at_client_obj_send(at_client_get_first(), buf, size) | ||||
| #define at_client_recv(buf, size, timeout)       at_client_obj_recv(at_client_get_first(), buf, size, timeout) | ||||
| #define at_set_end_sign(ch)                      at_obj_set_end_sign(at_client_get_first(), ch) | ||||
| #define at_set_urc_table(urc_table, table_sz)    at_obj_set_urc_table(at_client_get_first(), urc_table, table_sz) | ||||
|  | ||||
| #endif /* AT_USING_CLIENT */ | ||||
|  | ||||
| /* ========================== User port function ============================ */ | ||||
|  | ||||
| #ifdef AT_USING_SERVER | ||||
| /* AT server device reset */ | ||||
| void at_port_reset(void); | ||||
|  | ||||
| /* AT server device factory reset */ | ||||
| void at_port_factory_reset(void); | ||||
| #endif | ||||
|  | ||||
| #ifdef __cplusplus | ||||
| } | ||||
| #endif | ||||
|  | ||||
| #endif /* __AT_H__ */ | ||||
							
								
								
									
										27
									
								
								riscv/rtthread/components/net/at/include/at_log.h
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										27
									
								
								riscv/rtthread/components/net/at/include/at_log.h
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,27 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2018-10-12     armink       first version | ||||
|  */ | ||||
|  | ||||
| /* | ||||
|  * NOTE: DO NOT include this file on the header file. | ||||
|  */ | ||||
|  | ||||
| #ifndef LOG_TAG | ||||
| #define DBG_TAG               "at" | ||||
| #else | ||||
| #define DBG_TAG               LOG_TAG | ||||
| #endif /* LOG_TAG */ | ||||
|  | ||||
| #ifdef AT_DEBUG | ||||
| #define DBG_LVL               DBG_LOG | ||||
| #else | ||||
| #define DBG_LVL               DBG_INFO | ||||
| #endif /* AT_DEBUG */ | ||||
|  | ||||
| #include <rtdbg.h> | ||||
							
								
								
									
										121
									
								
								riscv/rtthread/components/net/at/src/at_base_cmd.c
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										121
									
								
								riscv/rtthread/components/net/at/src/at_base_cmd.c
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,121 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2018-04-01     armink       first version | ||||
|  * 2018-04-04     chenyong     add base commands | ||||
|  */ | ||||
|  | ||||
| #include <at.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
|  | ||||
| #include <rtdevice.h> | ||||
|  | ||||
| #ifdef AT_USING_SERVER | ||||
|  | ||||
| #define AT_ECHO_MODE_CLOSE             0 | ||||
| #define AT_ECHO_MODE_OPEN              1 | ||||
|  | ||||
| extern at_server_t at_get_server(void); | ||||
|  | ||||
| static at_result_t at_exec(void) | ||||
| { | ||||
|     return AT_RESULT_OK; | ||||
| } | ||||
| AT_CMD_EXPORT("AT", RT_NULL, RT_NULL, RT_NULL, RT_NULL, at_exec); | ||||
|  | ||||
| static at_result_t atz_exec(void) | ||||
| { | ||||
|     at_server_printfln("OK"); | ||||
|  | ||||
|     at_port_factory_reset(); | ||||
|  | ||||
|     return AT_RESULT_NULL; | ||||
| } | ||||
| AT_CMD_EXPORT("ATZ", RT_NULL, RT_NULL, RT_NULL, RT_NULL, atz_exec); | ||||
|  | ||||
| static at_result_t at_rst_exec(void) | ||||
| { | ||||
|     at_server_printfln("OK"); | ||||
|  | ||||
|     at_port_reset(); | ||||
|  | ||||
|     return AT_RESULT_NULL; | ||||
| } | ||||
| AT_CMD_EXPORT("AT+RST", RT_NULL, RT_NULL, RT_NULL, RT_NULL, at_rst_exec); | ||||
|  | ||||
| static at_result_t ate_setup(const char *args) | ||||
| { | ||||
|     int echo_mode = atoi(args); | ||||
|  | ||||
|     if(echo_mode == AT_ECHO_MODE_CLOSE || echo_mode == AT_ECHO_MODE_OPEN) | ||||
|     { | ||||
|         at_get_server()->echo_mode = echo_mode; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         return AT_RESULT_FAILE; | ||||
|     } | ||||
|  | ||||
|     return AT_RESULT_OK; | ||||
| } | ||||
| AT_CMD_EXPORT("ATE", "<value>", RT_NULL, RT_NULL, ate_setup, RT_NULL); | ||||
|  | ||||
| static at_result_t at_show_cmd_exec(void) | ||||
| { | ||||
|     extern void rt_at_server_print_all_cmd(void); | ||||
|  | ||||
|     rt_at_server_print_all_cmd(); | ||||
|  | ||||
|     return AT_RESULT_OK; | ||||
| } | ||||
| AT_CMD_EXPORT("AT&L", RT_NULL, RT_NULL, RT_NULL, RT_NULL, at_show_cmd_exec); | ||||
|  | ||||
| static at_result_t at_uart_query(void) | ||||
| { | ||||
|     struct rt_serial_device *serial = (struct rt_serial_device *)at_get_server()->device; | ||||
|  | ||||
|     at_server_printfln("AT+UART=%d,%d,%d,%d,%d", serial->config.baud_rate, serial->config.data_bits, | ||||
|             serial->config.stop_bits, serial->config.parity, 1); | ||||
|  | ||||
|     return AT_RESULT_OK; | ||||
| } | ||||
|  | ||||
| static at_result_t at_uart_setup(const char *args) | ||||
| { | ||||
|     struct serial_configure config = RT_SERIAL_CONFIG_DEFAULT; | ||||
|     int baudrate, databits, stopbits, parity, flow_control, argc; | ||||
|     const char *req_expr = "=%d,%d,%d,%d,%d"; | ||||
|  | ||||
|     argc = at_req_parse_args(args, req_expr, &baudrate, &databits, &stopbits, &parity, &flow_control); | ||||
|     if (argc != 5) | ||||
|     { | ||||
|         return AT_RESULT_PARSE_FAILE; | ||||
|     } | ||||
|  | ||||
|     at_server_printfln("UART baudrate : %d", baudrate); | ||||
|     at_server_printfln("UART databits : %d", databits); | ||||
|     at_server_printfln("UART stopbits : %d", stopbits); | ||||
|     at_server_printfln("UART parity   : %d", parity); | ||||
|     at_server_printfln("UART control  : %d", flow_control); | ||||
|  | ||||
|     config.baud_rate = baudrate; | ||||
|     config.data_bits = databits; | ||||
|     config.stop_bits = stopbits; | ||||
|     config.parity = parity; | ||||
|  | ||||
|     if(rt_device_control(at_get_server()->device, RT_DEVICE_CTRL_CONFIG, &config) != RT_EOK) | ||||
|     { | ||||
|         return AT_RESULT_FAILE; | ||||
|     } | ||||
|  | ||||
|     return AT_RESULT_OK; | ||||
| } | ||||
|  | ||||
| AT_CMD_EXPORT("AT+UART", "=<baudrate>,<databits>,<stopbits>,<parity>,<flow_control>", RT_NULL, at_uart_query, at_uart_setup, RT_NULL); | ||||
|  | ||||
| #endif /* AT_USING_SERVER */ | ||||
							
								
								
									
										345
									
								
								riscv/rtthread/components/net/at/src/at_cli.c
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										345
									
								
								riscv/rtthread/components/net/at/src/at_cli.c
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,345 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2018-04-02     armink       first version | ||||
|  */ | ||||
|  | ||||
| #include <at.h> | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
| #include <stdint.h> | ||||
| #include <rtthread.h> | ||||
| #include <rtdevice.h> | ||||
| #include <rthw.h> | ||||
|  | ||||
| #ifdef AT_USING_CLI | ||||
|  | ||||
| #define AT_CLI_FIFO_SIZE                      256 | ||||
|  | ||||
| static struct rt_semaphore console_rx_notice; | ||||
| static struct rt_ringbuffer *console_rx_fifo = RT_NULL; | ||||
| static rt_err_t (*odev_rx_ind)(rt_device_t dev, rt_size_t size) = RT_NULL; | ||||
|  | ||||
| #ifdef AT_USING_CLIENT | ||||
| static struct rt_semaphore client_rx_notice; | ||||
| static struct rt_ringbuffer *client_rx_fifo = RT_NULL; | ||||
| #endif | ||||
|  | ||||
| static char console_getchar(void) | ||||
| { | ||||
|     char ch; | ||||
|  | ||||
|     rt_sem_take(&console_rx_notice, RT_WAITING_FOREVER); | ||||
|     rt_ringbuffer_getchar(console_rx_fifo, (rt_uint8_t *)&ch); | ||||
|  | ||||
|     return ch; | ||||
| } | ||||
|  | ||||
| static rt_err_t console_getchar_rx_ind(rt_device_t dev, rt_size_t size) | ||||
| { | ||||
|     uint8_t ch; | ||||
|     rt_size_t i; | ||||
|  | ||||
|     for (i = 0; i < size; i++) | ||||
|     { | ||||
|         /* read a char */ | ||||
|         if (rt_device_read(dev, 0, &ch, 1)) | ||||
|         { | ||||
|             rt_ringbuffer_put_force(console_rx_fifo, &ch, 1); | ||||
|             rt_sem_release(&console_rx_notice); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| void at_cli_init(void) | ||||
| { | ||||
|     rt_base_t level; | ||||
|     rt_device_t console; | ||||
|  | ||||
|     rt_sem_init(&console_rx_notice, "cli_c", 0, RT_IPC_FLAG_FIFO); | ||||
|  | ||||
|     /* create RX FIFO */ | ||||
|     console_rx_fifo = rt_ringbuffer_create(AT_CLI_FIFO_SIZE); | ||||
|     /* created must success */ | ||||
|     RT_ASSERT(console_rx_fifo); | ||||
|  | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|     console = rt_console_get_device(); | ||||
|     if (console) | ||||
|     { | ||||
|         /* backup RX indicate */ | ||||
|         odev_rx_ind = console->rx_indicate; | ||||
|         rt_device_set_rx_indicate(console, console_getchar_rx_ind); | ||||
|     } | ||||
|  | ||||
|     rt_hw_interrupt_enable(level); | ||||
| } | ||||
|  | ||||
| void at_cli_deinit(void) | ||||
| { | ||||
|     rt_base_t level; | ||||
|     rt_device_t console; | ||||
|  | ||||
|     level = rt_hw_interrupt_disable(); | ||||
|     console = rt_console_get_device(); | ||||
|     if (console && odev_rx_ind) | ||||
|     { | ||||
|         /* restore RX indicate */ | ||||
|         rt_device_set_rx_indicate(console, odev_rx_ind); | ||||
|     } | ||||
|     rt_hw_interrupt_enable(level); | ||||
|  | ||||
|     rt_sem_detach(&console_rx_notice); | ||||
|     rt_ringbuffer_destroy(console_rx_fifo); | ||||
| } | ||||
|  | ||||
| #ifdef AT_USING_SERVER | ||||
| static rt_err_t at_server_console_getchar(struct at_server *server, char *ch, rt_int32_t timeout) | ||||
| { | ||||
|     *ch = console_getchar(); | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| static void server_cli_parser(void) | ||||
| { | ||||
|     extern at_server_t at_get_server(void); | ||||
|  | ||||
|     at_server_t server = at_get_server(); | ||||
|     rt_base_t level; | ||||
|     static rt_device_t device_bak; | ||||
|     static rt_err_t (*getchar_bak)(struct at_server *server, char *ch, rt_int32_t timeout); | ||||
|  | ||||
|     /* backup server device and getchar function */ | ||||
|     { | ||||
|         level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|         device_bak = server->device; | ||||
|         getchar_bak = server->get_char; | ||||
|  | ||||
|         /* setup server device as console device */ | ||||
|         server->device = rt_console_get_device(); | ||||
|         server->get_char = at_server_console_getchar; | ||||
|  | ||||
|         rt_hw_interrupt_enable(level); | ||||
|     } | ||||
|  | ||||
|     if (server) | ||||
|     { | ||||
|         rt_kprintf("======== Welcome to using RT-Thread AT command server cli ========\n"); | ||||
|         rt_kprintf("Input your at command for test server. Press 'ESC' to exit.\n"); | ||||
|         server->parser_entry(server); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         rt_kprintf("AT client not initialized\n"); | ||||
|     } | ||||
|  | ||||
|     /* restore server device and getchar function */ | ||||
|     { | ||||
|         level = rt_hw_interrupt_disable(); | ||||
|  | ||||
|         server->device = device_bak; | ||||
|         server->get_char = getchar_bak; | ||||
|  | ||||
|         rt_hw_interrupt_enable(level); | ||||
|     } | ||||
| } | ||||
| #endif /* AT_USING_SERVER */ | ||||
|  | ||||
| #ifdef AT_USING_CLIENT | ||||
| static char client_getchar(void) | ||||
| { | ||||
|     char ch; | ||||
|  | ||||
|     rt_sem_take(&client_rx_notice, RT_WAITING_FOREVER); | ||||
|     rt_ringbuffer_getchar(client_rx_fifo, (rt_uint8_t *)&ch); | ||||
|  | ||||
|     return ch; | ||||
| } | ||||
|  | ||||
| static void at_client_entry(void *param) | ||||
| { | ||||
|     char ch; | ||||
|  | ||||
|     while(1) | ||||
|     { | ||||
|         ch = client_getchar(); | ||||
|         rt_kprintf("%c", ch); | ||||
|     } | ||||
| } | ||||
|  | ||||
| static rt_err_t client_getchar_rx_ind(rt_device_t dev, rt_size_t size) | ||||
| { | ||||
|     uint8_t ch; | ||||
|     rt_size_t i; | ||||
|  | ||||
|     for (i = 0; i < size; i++) | ||||
|     { | ||||
|         /* read a char */ | ||||
|         if (rt_device_read(dev, 0, &ch, 1)) | ||||
|         { | ||||
|             rt_ringbuffer_put_force(client_rx_fifo, &ch, 1); | ||||
|             rt_sem_release(&client_rx_notice); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
| static void client_cli_parser(at_client_t  client) | ||||
| { | ||||
| #define ESC_KEY                 0x1B | ||||
| #define BACKSPACE_KEY           0x08 | ||||
| #define DELECT_KEY              0x7F | ||||
|  | ||||
|     char ch; | ||||
|     char cur_line[FINSH_CMD_SIZE] = { 0 }; | ||||
|     rt_size_t cur_line_len = 0; | ||||
|     static rt_err_t (*client_odev_rx_ind)(rt_device_t dev, rt_size_t size) = RT_NULL; | ||||
|     rt_base_t level; | ||||
|     rt_thread_t at_client; | ||||
|     at_status_t client_odev_status; | ||||
|  | ||||
|     if (client) | ||||
|     { | ||||
|         /* backup client status */ | ||||
|         { | ||||
|             client_odev_status = client->status; | ||||
|             client->status = AT_STATUS_CLI; | ||||
|         } | ||||
|  | ||||
|         rt_sem_init(&client_rx_notice, "cli_r", 0, RT_IPC_FLAG_FIFO); | ||||
|         client_rx_fifo = rt_ringbuffer_create(AT_CLI_FIFO_SIZE); | ||||
|  | ||||
|         /* backup client device RX indicate */ | ||||
|         { | ||||
|             level = rt_hw_interrupt_disable(); | ||||
|             client_odev_rx_ind = client->device->rx_indicate; | ||||
|             rt_device_set_rx_indicate(client->device, client_getchar_rx_ind); | ||||
|             rt_hw_interrupt_enable(level); | ||||
|         } | ||||
|  | ||||
|         at_client = rt_thread_create("at_cli", at_client_entry, RT_NULL, 512, 8, 8); | ||||
|         if (client_rx_fifo && at_client) | ||||
|         { | ||||
|             rt_kprintf("======== Welcome to using RT-Thread AT command client cli ========\n"); | ||||
|             rt_kprintf("Cli will forward your command to server port(%s). Press 'ESC' to exit.\n", client->device->parent.name); | ||||
|             rt_thread_startup(at_client); | ||||
|             /* process user input */ | ||||
|             while (ESC_KEY != (ch = console_getchar())) | ||||
|             { | ||||
|                 if (ch == BACKSPACE_KEY || ch == DELECT_KEY) | ||||
|                 { | ||||
|                     if (cur_line_len) | ||||
|                     { | ||||
|                         cur_line[--cur_line_len] = 0; | ||||
|                         rt_kprintf("\b \b"); | ||||
|                     } | ||||
|                     continue; | ||||
|                 } | ||||
|                 else if (ch == '\r' || ch == '\n') | ||||
|                 { | ||||
|                     /* execute a AT request */ | ||||
|                     if (cur_line_len) | ||||
|                     { | ||||
|                         rt_kprintf("\n"); | ||||
|                         at_obj_exec_cmd(client, RT_NULL, "%.*s", cur_line_len, cur_line); | ||||
|                     } | ||||
|                     cur_line_len = 0; | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     if(cur_line_len >= FINSH_CMD_SIZE) | ||||
|                     { | ||||
|                         continue; | ||||
|                     } | ||||
|                     rt_kprintf("%c", ch); | ||||
|                     cur_line[cur_line_len++] = ch; | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             /* restore client status */ | ||||
|             client->status = client_odev_status; | ||||
|  | ||||
|             /* restore client device RX indicate */ | ||||
|             { | ||||
|                 level = rt_hw_interrupt_disable(); | ||||
|                 rt_device_set_rx_indicate(client->device, client_odev_rx_ind); | ||||
|                 rt_hw_interrupt_enable(level); | ||||
|             } | ||||
|  | ||||
|             rt_thread_delete(at_client); | ||||
|             rt_sem_detach(&client_rx_notice); | ||||
|             rt_ringbuffer_destroy(client_rx_fifo); | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             rt_kprintf("No mem for AT cli client\n"); | ||||
|         } | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         rt_kprintf("AT client not initialized\n"); | ||||
|     } | ||||
| } | ||||
| #endif /* AT_USING_CLIENT */ | ||||
|  | ||||
| static void at(int argc, char **argv) | ||||
| { | ||||
|  | ||||
|     if (argc != 2 && argc != 3) | ||||
|     { | ||||
|         rt_kprintf("Please input '<server|client [dev_name]>' \n"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     at_cli_init(); | ||||
|  | ||||
|     if (!strcmp(argv[1], "server")) | ||||
|     { | ||||
| #ifdef AT_USING_SERVER | ||||
|         server_cli_parser(); | ||||
| #else | ||||
|         rt_kprintf("Not support AT server, please check your configure!\n"); | ||||
| #endif /* AT_USING_SERVER */ | ||||
|     } | ||||
|     else if (!strcmp(argv[1], "client")) | ||||
|     { | ||||
| #ifdef AT_USING_CLIENT | ||||
|         at_client_t client = RT_NULL; | ||||
|  | ||||
|         if (argc == 2) | ||||
|         { | ||||
|             client_cli_parser(at_client_get_first()); | ||||
|         } | ||||
|         else if (argc == 3) | ||||
|         { | ||||
|             client = at_client_get(argv[2]); | ||||
|             if (client == RT_NULL) | ||||
|             { | ||||
|                 rt_kprintf("input AT client device name(%s) error.\n", argv[2]); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 client_cli_parser(client); | ||||
|             } | ||||
|         } | ||||
| #else | ||||
|         rt_kprintf("Not support AT client, please check your configure!\n"); | ||||
| #endif /* AT_USING_CLIENT */ | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         rt_kprintf("Please input '<server|client [dev_name]>' \n"); | ||||
|     } | ||||
|  | ||||
|     at_cli_deinit(); | ||||
| } | ||||
| MSH_CMD_EXPORT(at, RT-Thread AT component cli: at <server|client [dev_name]>); | ||||
|  | ||||
| #endif /* AT_USING_CLI */ | ||||
							
								
								
									
										994
									
								
								riscv/rtthread/components/net/at/src/at_client.c
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										994
									
								
								riscv/rtthread/components/net/at/src/at_client.c
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,994 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2023, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2018-03-30     chenyong     first version | ||||
|  * 2018-04-12     chenyong     add client implement | ||||
|  * 2018-08-17     chenyong     multiple client support | ||||
|  * 2021-03-17     Meco Man     fix a buf of leaking memory | ||||
|  * 2021-07-14     Sszl         fix a buf of leaking memory | ||||
|  */ | ||||
|  | ||||
| #include <at.h> | ||||
| #include <stdio.h> | ||||
| #include <stdlib.h> | ||||
| #include <string.h> | ||||
|  | ||||
| #define LOG_TAG              "at.clnt" | ||||
| #include <at_log.h> | ||||
|  | ||||
| #ifdef AT_USING_CLIENT | ||||
|  | ||||
| #define AT_RESP_END_OK                 "OK" | ||||
| #define AT_RESP_END_ERROR              "ERROR" | ||||
| #define AT_RESP_END_FAIL               "FAIL" | ||||
| #define AT_END_CR_LF                   "\r\n" | ||||
|  | ||||
| static struct at_client at_client_table[AT_CLIENT_NUM_MAX] = { 0 }; | ||||
|  | ||||
| extern rt_size_t at_utils_send(rt_device_t dev, | ||||
|                                rt_off_t    pos, | ||||
|                                const void *buffer, | ||||
|                                rt_size_t   size); | ||||
| extern rt_size_t at_vprintfln(rt_device_t device, char *send_buf, rt_size_t buf_size, const char *format, va_list args); | ||||
| extern void at_print_raw_cmd(const char *type, const char *cmd, rt_size_t size); | ||||
|  | ||||
| /** | ||||
|  * Create response object. | ||||
|  * | ||||
|  * @param buf_size the maximum response buffer size | ||||
|  * @param line_num the number of setting response lines | ||||
|  *         = 0: the response data will auto return when received 'OK' or 'ERROR' | ||||
|  *        != 0: the response data will return when received setting lines number data | ||||
|  * @param timeout the maximum response time | ||||
|  * | ||||
|  * @return != RT_NULL: response object | ||||
|  *          = RT_NULL: no memory | ||||
|  */ | ||||
| at_response_t at_create_resp(rt_size_t buf_size, rt_size_t line_num, rt_int32_t timeout) | ||||
| { | ||||
|     at_response_t resp = RT_NULL; | ||||
|  | ||||
|     resp = (at_response_t) rt_calloc(1, sizeof(struct at_response)); | ||||
|     if (resp == RT_NULL) | ||||
|     { | ||||
|         LOG_E("AT create response object failed! No memory for response object!"); | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     resp->buf = (char *) rt_calloc(1, buf_size); | ||||
|     if (resp->buf == RT_NULL) | ||||
|     { | ||||
|         LOG_E("AT create response object failed! No memory for response buffer!"); | ||||
|         rt_free(resp); | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     resp->buf_size = buf_size; | ||||
|     resp->line_num = line_num; | ||||
|     resp->line_counts = 0; | ||||
|     resp->timeout = timeout; | ||||
|  | ||||
|     return resp; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Delete and free response object. | ||||
|  * | ||||
|  * @param resp response object | ||||
|  */ | ||||
| void at_delete_resp(at_response_t resp) | ||||
| { | ||||
|     if (resp && resp->buf) | ||||
|     { | ||||
|         rt_free(resp->buf); | ||||
|     } | ||||
|  | ||||
|     if (resp) | ||||
|     { | ||||
|         rt_free(resp); | ||||
|         resp = RT_NULL; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Set response object information | ||||
|  * | ||||
|  * @param resp response object | ||||
|  * @param buf_size the maximum response buffer size | ||||
|  * @param line_num the number of setting response lines | ||||
|  *         = 0: the response data will auto return when received 'OK' or 'ERROR' | ||||
|  *        != 0: the response data will return when received setting lines number data | ||||
|  * @param timeout the maximum response time | ||||
|  * | ||||
|  * @return  != RT_NULL: response object | ||||
|  *           = RT_NULL: no memory | ||||
|  */ | ||||
| at_response_t at_resp_set_info(at_response_t resp, rt_size_t buf_size, rt_size_t line_num, rt_int32_t timeout) | ||||
| { | ||||
|     char *p_temp; | ||||
|     RT_ASSERT(resp); | ||||
|  | ||||
|     if (resp->buf_size != buf_size) | ||||
|     { | ||||
|         resp->buf_size = buf_size; | ||||
|  | ||||
|         p_temp = (char *) rt_realloc(resp->buf, buf_size); | ||||
|         if (p_temp == RT_NULL) | ||||
|         { | ||||
|             LOG_D("No memory for realloc response buffer size(%d).", buf_size); | ||||
|             return RT_NULL; | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             resp->buf = p_temp; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     resp->line_num = line_num; | ||||
|     resp->timeout = timeout; | ||||
|  | ||||
|     return resp; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get one line AT response buffer by line number. | ||||
|  * | ||||
|  * @param resp response object | ||||
|  * @param resp_line line number, start from '1' | ||||
|  * | ||||
|  * @return != RT_NULL: response line buffer | ||||
|  *          = RT_NULL: input response line error | ||||
|  */ | ||||
| const char *at_resp_get_line(at_response_t resp, rt_size_t resp_line) | ||||
| { | ||||
|     char *resp_buf = resp->buf; | ||||
|     rt_size_t line_num = 1; | ||||
|  | ||||
|     RT_ASSERT(resp); | ||||
|  | ||||
|     if (resp_line > resp->line_counts || resp_line <= 0) | ||||
|     { | ||||
|         LOG_E("AT response get line failed! Input response line(%d) error!", resp_line); | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     for (line_num = 1; line_num <= resp->line_counts; line_num++) | ||||
|     { | ||||
|         if (resp_line == line_num) | ||||
|         { | ||||
|             return resp_buf; | ||||
|         } | ||||
|  | ||||
|         resp_buf += strlen(resp_buf) + 1; | ||||
|     } | ||||
|  | ||||
|     return RT_NULL; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get one line AT response buffer by keyword | ||||
|  * | ||||
|  * @param resp response object | ||||
|  * @param keyword query keyword | ||||
|  * | ||||
|  * @return != RT_NULL: response line buffer | ||||
|  *          = RT_NULL: no matching data | ||||
|  */ | ||||
| const char *at_resp_get_line_by_kw(at_response_t resp, const char *keyword) | ||||
| { | ||||
|     char *resp_buf = resp->buf; | ||||
|     rt_size_t line_num = 1; | ||||
|  | ||||
|     RT_ASSERT(resp); | ||||
|     RT_ASSERT(keyword); | ||||
|  | ||||
|     for (line_num = 1; line_num <= resp->line_counts; line_num++) | ||||
|     { | ||||
|         if (strstr(resp_buf, keyword)) | ||||
|         { | ||||
|             return resp_buf; | ||||
|         } | ||||
|  | ||||
|         resp_buf += strlen(resp_buf) + 1; | ||||
|     } | ||||
|  | ||||
|     return RT_NULL; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get and parse AT response buffer arguments by line number. | ||||
|  * | ||||
|  * @param resp response object | ||||
|  * @param resp_line line number, start from '1' | ||||
|  * @param resp_expr response buffer expression | ||||
|  * | ||||
|  * @return -1 : input response line number error or get line buffer error | ||||
|  *          0 : parsed without match | ||||
|  *         >0 : the number of arguments successfully parsed | ||||
|  */ | ||||
| int at_resp_parse_line_args(at_response_t resp, rt_size_t resp_line, const char *resp_expr, ...) | ||||
| { | ||||
|     va_list args; | ||||
|     int resp_args_num = 0; | ||||
|     const char *resp_line_buf = RT_NULL; | ||||
|  | ||||
|     RT_ASSERT(resp); | ||||
|     RT_ASSERT(resp_expr); | ||||
|  | ||||
|     if ((resp_line_buf = at_resp_get_line(resp, resp_line)) == RT_NULL) | ||||
|     { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     va_start(args, resp_expr); | ||||
|  | ||||
|     resp_args_num = vsscanf(resp_line_buf, resp_expr, args); | ||||
|  | ||||
|     va_end(args); | ||||
|  | ||||
|     return resp_args_num; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Get and parse AT response buffer arguments by keyword. | ||||
|  * | ||||
|  * @param resp response object | ||||
|  * @param keyword query keyword | ||||
|  * @param resp_expr response buffer expression | ||||
|  * | ||||
|  * @return -1 : input keyword error or get line buffer error | ||||
|  *          0 : parsed without match | ||||
|  *         >0 : the number of arguments successfully parsed | ||||
|  */ | ||||
| int at_resp_parse_line_args_by_kw(at_response_t resp, const char *keyword, const char *resp_expr, ...) | ||||
| { | ||||
|     va_list args; | ||||
|     int resp_args_num = 0; | ||||
|     const char *resp_line_buf = RT_NULL; | ||||
|  | ||||
|     RT_ASSERT(resp); | ||||
|     RT_ASSERT(resp_expr); | ||||
|  | ||||
|     if ((resp_line_buf = at_resp_get_line_by_kw(resp, keyword)) == RT_NULL) | ||||
|     { | ||||
|         return -1; | ||||
|     } | ||||
|  | ||||
|     va_start(args, resp_expr); | ||||
|  | ||||
|     resp_args_num = vsscanf(resp_line_buf, resp_expr, args); | ||||
|  | ||||
|     va_end(args); | ||||
|  | ||||
|     return resp_args_num; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Send commands to AT server and wait response. | ||||
|  * | ||||
|  * @param client current AT client object | ||||
|  * @param resp AT response object, using RT_NULL when you don't care response | ||||
|  * @param cmd_expr AT commands expression | ||||
|  * | ||||
|  * @return 0 : success | ||||
|  *        -1 : response status error | ||||
|  *        -2 : wait timeout | ||||
|  *        -7 : enter AT CLI mode | ||||
|  */ | ||||
| int at_obj_exec_cmd(at_client_t client, at_response_t resp, const char *cmd_expr, ...) | ||||
| { | ||||
|     va_list args; | ||||
|     rt_err_t result = RT_EOK; | ||||
|  | ||||
|     RT_ASSERT(cmd_expr); | ||||
|  | ||||
|     if (client == RT_NULL) | ||||
|     { | ||||
|         LOG_E("input AT Client object is NULL, please create or get AT Client object!"); | ||||
|         return -RT_ERROR; | ||||
|     } | ||||
|  | ||||
|     /* check AT CLI mode */ | ||||
|     if (client->status == AT_STATUS_CLI && resp) | ||||
|     { | ||||
|         return -RT_EBUSY; | ||||
|     } | ||||
|  | ||||
|     rt_mutex_take(client->lock, RT_WAITING_FOREVER); | ||||
|  | ||||
|     client->resp_status = AT_RESP_OK; | ||||
|  | ||||
|     if (resp != RT_NULL) | ||||
|     { | ||||
|         resp->buf_len = 0; | ||||
|         resp->line_counts = 0; | ||||
|     } | ||||
|  | ||||
|     client->resp = resp; | ||||
|     rt_sem_control(client->resp_notice, RT_IPC_CMD_RESET, RT_NULL); | ||||
|  | ||||
|     va_start(args, cmd_expr); | ||||
|     client->last_cmd_len = at_vprintfln(client->device, client->send_buf, client->send_bufsz, cmd_expr, args); | ||||
|     if (client->last_cmd_len > 2) | ||||
|     { | ||||
|         client->last_cmd_len -= 2; /* "\r\n" */ | ||||
|     } | ||||
|     va_end(args); | ||||
|  | ||||
|     if (resp != RT_NULL) | ||||
|     { | ||||
|         if (rt_sem_take(client->resp_notice, resp->timeout) != RT_EOK) | ||||
|         { | ||||
|             LOG_W("execute command (%.*s) timeout (%d ticks)!", client->last_cmd_len, client->send_buf, resp->timeout); | ||||
|             client->resp_status = AT_RESP_TIMEOUT; | ||||
|             result = -RT_ETIMEOUT; | ||||
|         } | ||||
|         else if (client->resp_status != AT_RESP_OK) | ||||
|         { | ||||
|             LOG_E("execute command (%.*s) failed!", client->last_cmd_len, client->send_buf); | ||||
|             result = -RT_ERROR; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     client->resp = RT_NULL; | ||||
|  | ||||
|     rt_mutex_release(client->lock); | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Waiting for connection to external devices. | ||||
|  * | ||||
|  * @param client current AT client object | ||||
|  * @param timeout millisecond for timeout | ||||
|  * | ||||
|  * @return 0 : success | ||||
|  *        -2 : timeout | ||||
|  *        -5 : no memory | ||||
|  */ | ||||
| int at_client_obj_wait_connect(at_client_t client, rt_uint32_t timeout) | ||||
| { | ||||
|     rt_err_t result = RT_EOK; | ||||
|     at_response_t resp = RT_NULL; | ||||
|     rt_tick_t start_time = 0; | ||||
|     char *client_name = client->device->parent.name; | ||||
|  | ||||
|     if (client == RT_NULL) | ||||
|     { | ||||
|         LOG_E("input AT client object is NULL, please create or get AT Client object!"); | ||||
|         return -RT_ERROR; | ||||
|     } | ||||
|  | ||||
|     resp = at_create_resp(64, 0, rt_tick_from_millisecond(300)); | ||||
|     if (resp == RT_NULL) | ||||
|     { | ||||
|         LOG_E("no memory for AT client(%s) response object.", client_name); | ||||
|         return -RT_ENOMEM; | ||||
|     } | ||||
|  | ||||
|     rt_mutex_take(client->lock, RT_WAITING_FOREVER); | ||||
|     client->resp = resp; | ||||
|     rt_sem_control(client->resp_notice, RT_IPC_CMD_RESET, RT_NULL); | ||||
|  | ||||
|     start_time = rt_tick_get(); | ||||
|  | ||||
|     while (1) | ||||
|     { | ||||
|         /* Check whether it is timeout */ | ||||
|         if (rt_tick_get() - start_time > rt_tick_from_millisecond(timeout)) | ||||
|         { | ||||
|             LOG_E("wait AT client(%s) connect timeout(%d tick).", client_name, timeout); | ||||
|             result = -RT_ETIMEOUT; | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         /* Check whether it is already connected */ | ||||
|         resp->buf_len = 0; | ||||
|         resp->line_counts = 0; | ||||
|         at_utils_send(client->device, 0, "AT\r\n", 4); | ||||
|  | ||||
|         if (rt_sem_take(client->resp_notice, resp->timeout) == RT_EOK) | ||||
|             break; | ||||
|     } | ||||
|  | ||||
|     at_delete_resp(resp); | ||||
|  | ||||
|     client->resp = RT_NULL; | ||||
|  | ||||
|     rt_mutex_release(client->lock); | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Send data to AT server, send data don't have end sign(eg: \r\n). | ||||
|  * | ||||
|  * @param client current AT client object | ||||
|  * @param buf   send data buffer | ||||
|  * @param size  send fixed data size | ||||
|  * | ||||
|  * @return >0: send data size | ||||
|  *         =0: send failed | ||||
|  */ | ||||
| rt_size_t at_client_obj_send(at_client_t client, const char *buf, rt_size_t size) | ||||
| { | ||||
|     rt_size_t len; | ||||
|  | ||||
|     RT_ASSERT(buf); | ||||
|  | ||||
|     if (client == RT_NULL) | ||||
|     { | ||||
|         LOG_E("input AT Client object is NULL, please create or get AT Client object!"); | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
| #ifdef AT_PRINT_RAW_CMD | ||||
|     at_print_raw_cmd("sendline", buf, size); | ||||
| #endif | ||||
|  | ||||
|     rt_mutex_take(client->lock, RT_WAITING_FOREVER); | ||||
|  | ||||
|     len = at_utils_send(client->device, 0, buf, size); | ||||
|  | ||||
|     rt_mutex_release(client->lock); | ||||
|  | ||||
|     return len; | ||||
| } | ||||
|  | ||||
| static rt_err_t at_client_getchar(at_client_t client, char *ch, rt_int32_t timeout) | ||||
| { | ||||
|     rt_err_t result = RT_EOK; | ||||
|  | ||||
|     while (rt_device_read(client->device, 0, ch, 1) == 0) | ||||
|     { | ||||
|         result = rt_sem_take(client->rx_notice, rt_tick_from_millisecond(timeout)); | ||||
|         if (result != RT_EOK) | ||||
|         { | ||||
|             return result; | ||||
|         } | ||||
|  | ||||
|         rt_sem_control(client->rx_notice, RT_IPC_CMD_RESET, RT_NULL); | ||||
|     } | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * AT client receive fixed-length data. | ||||
|  * | ||||
|  * @param client current AT client object | ||||
|  * @param buf   receive data buffer | ||||
|  * @param size  receive fixed data size | ||||
|  * @param timeout  receive data timeout (ms) | ||||
|  * | ||||
|  * @note this function can only be used in execution function of URC data | ||||
|  * | ||||
|  * @return >0: receive data size | ||||
|  *         =0: receive failed | ||||
|  */ | ||||
| rt_size_t at_client_obj_recv(at_client_t client, char *buf, rt_size_t size, rt_int32_t timeout) | ||||
| { | ||||
|     rt_size_t len = 0; | ||||
|  | ||||
|     RT_ASSERT(buf); | ||||
|  | ||||
|     if (client == RT_NULL) | ||||
|     { | ||||
|         LOG_E("input AT Client object is NULL, please create or get AT Client object!"); | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     while (size) | ||||
|     { | ||||
|         rt_size_t read_len; | ||||
|  | ||||
|         rt_sem_control(client->rx_notice, RT_IPC_CMD_RESET, RT_NULL); | ||||
|  | ||||
|         read_len = rt_device_read(client->device, 0, buf + len, size); | ||||
|         if (read_len > 0) | ||||
|         { | ||||
|             len += read_len; | ||||
|             size -= read_len; | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             if (rt_sem_take(client->rx_notice, rt_tick_from_millisecond(timeout)) != RT_EOK) | ||||
|                 break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
| #ifdef AT_PRINT_RAW_CMD | ||||
|     at_print_raw_cmd("urc_recv", buf, len); | ||||
| #endif | ||||
|  | ||||
|     return len; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  *  AT client set end sign. | ||||
|  * | ||||
|  * @param client current AT client object | ||||
|  * @param ch the end sign, can not be used when it is '\0' | ||||
|  */ | ||||
| void at_obj_set_end_sign(at_client_t client, char ch) | ||||
| { | ||||
|     if (client == RT_NULL) | ||||
|     { | ||||
|         LOG_E("input AT Client object is NULL, please create or get AT Client object!"); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     client->end_sign = ch; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * set URC(Unsolicited Result Code) table | ||||
|  * | ||||
|  * @param client current AT client object | ||||
|  * @param table URC table | ||||
|  * @param size table size | ||||
|  */ | ||||
| int at_obj_set_urc_table(at_client_t client, const struct at_urc *urc_table, rt_size_t table_sz) | ||||
| { | ||||
|     rt_size_t idx; | ||||
|  | ||||
|     if (client == RT_NULL) | ||||
|     { | ||||
|         LOG_E("input AT Client object is NULL, please create or get AT Client object!"); | ||||
|         return -RT_ERROR; | ||||
|     } | ||||
|  | ||||
|     for (idx = 0; idx < table_sz; idx++) | ||||
|     { | ||||
|         RT_ASSERT(urc_table[idx].cmd_prefix); | ||||
|         RT_ASSERT(urc_table[idx].cmd_suffix); | ||||
|     } | ||||
|  | ||||
|     if (client->urc_table_size == 0) | ||||
|     { | ||||
|         client->urc_table = (struct at_urc_table *) rt_calloc(1, sizeof(struct at_urc_table)); | ||||
|         if (client->urc_table == RT_NULL) | ||||
|         { | ||||
|             return -RT_ENOMEM; | ||||
|         } | ||||
|  | ||||
|         client->urc_table[0].urc = urc_table; | ||||
|         client->urc_table[0].urc_size = table_sz; | ||||
|         client->urc_table_size++; | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         struct at_urc_table *new_urc_table = RT_NULL; | ||||
|  | ||||
|         /* realloc urc table space */ | ||||
|         new_urc_table = (struct at_urc_table *) rt_realloc(client->urc_table,client->urc_table_size * sizeof(struct at_urc_table) + sizeof(struct at_urc_table)); | ||||
|         if (new_urc_table == RT_NULL) | ||||
|         { | ||||
|             return -RT_ENOMEM; | ||||
|         } | ||||
|         client->urc_table = new_urc_table; | ||||
|         client->urc_table[client->urc_table_size].urc = urc_table; | ||||
|         client->urc_table[client->urc_table_size].urc_size = table_sz; | ||||
|         client->urc_table_size++; | ||||
|  | ||||
|     } | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * get AT client object by AT device name. | ||||
|  * | ||||
|  * @dev_name AT client device name | ||||
|  * | ||||
|  * @return AT client object | ||||
|  */ | ||||
| at_client_t at_client_get(const char *dev_name) | ||||
| { | ||||
|     int idx = 0; | ||||
|  | ||||
|     RT_ASSERT(dev_name); | ||||
|  | ||||
|     for (idx = 0; idx < AT_CLIENT_NUM_MAX; idx++) | ||||
|     { | ||||
|         if (at_client_table[idx].device && | ||||
|         (rt_strcmp(at_client_table[idx].device->parent.name, dev_name) == 0)) | ||||
|         { | ||||
|             return &at_client_table[idx]; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return RT_NULL; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * get first AT client object in the table. | ||||
|  * | ||||
|  * @return AT client object | ||||
|  */ | ||||
| at_client_t at_client_get_first(void) | ||||
| { | ||||
|     if (at_client_table[0].device == RT_NULL) | ||||
|     { | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     return &at_client_table[0]; | ||||
| } | ||||
|  | ||||
| static const struct at_urc *get_urc_obj(at_client_t client) | ||||
| { | ||||
|     rt_size_t i, j, prefix_len, suffix_len; | ||||
|     rt_size_t bufsz; | ||||
|     char *buffer = RT_NULL; | ||||
|     const struct at_urc *urc = RT_NULL; | ||||
|     struct at_urc_table *urc_table = RT_NULL; | ||||
|  | ||||
|     if (client->urc_table == RT_NULL) | ||||
|     { | ||||
|         return RT_NULL; | ||||
|     } | ||||
|  | ||||
|     buffer = client->recv_line_buf; | ||||
|     bufsz = client->recv_line_len; | ||||
|  | ||||
|     for (i = 0; i < client->urc_table_size; i++) | ||||
|     { | ||||
|         for (j = 0; j < client->urc_table[i].urc_size; j++) | ||||
|         { | ||||
|             urc_table = client->urc_table + i; | ||||
|             urc = urc_table->urc + j; | ||||
|  | ||||
|             prefix_len = rt_strlen(urc->cmd_prefix); | ||||
|             suffix_len = rt_strlen(urc->cmd_suffix); | ||||
|             if (bufsz < prefix_len + suffix_len) | ||||
|             { | ||||
|                 continue; | ||||
|             } | ||||
|             if ((prefix_len ? !rt_strncmp(buffer, urc->cmd_prefix, prefix_len) : 1) | ||||
|                     && (suffix_len ? !rt_strncmp(buffer + bufsz - suffix_len, urc->cmd_suffix, suffix_len) : 1)) | ||||
|             { | ||||
|                 return urc; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return RT_NULL; | ||||
| } | ||||
|  | ||||
| static int at_recv_readline(at_client_t client) | ||||
| { | ||||
|     char ch = 0, last_ch = 0; | ||||
|     rt_bool_t is_full = RT_FALSE; | ||||
|  | ||||
|     rt_memset(client->recv_line_buf, 0x00, client->recv_bufsz); | ||||
|     client->recv_line_len = 0; | ||||
|  | ||||
|     while (1) | ||||
|     { | ||||
|         at_client_getchar(client, &ch, RT_WAITING_FOREVER); | ||||
|  | ||||
|         if (client->recv_line_len < client->recv_bufsz) | ||||
|         { | ||||
|             client->recv_line_buf[client->recv_line_len++] = ch; | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             is_full = RT_TRUE; | ||||
|         } | ||||
|  | ||||
|         /* is newline or URC data */ | ||||
|         if ((client->urc = get_urc_obj(client)) != RT_NULL || (ch == '\n' && last_ch == '\r') | ||||
|                 || (client->end_sign != 0 && ch == client->end_sign)) | ||||
|         { | ||||
|             if (is_full) | ||||
|             { | ||||
|                 LOG_E("read line failed. The line data length is out of buffer size(%d)!", client->recv_bufsz); | ||||
|                 rt_memset(client->recv_line_buf, 0x00, client->recv_bufsz); | ||||
|                 client->recv_line_len = 0; | ||||
|                 return -RT_EFULL; | ||||
|             } | ||||
|             break; | ||||
|         } | ||||
|         last_ch = ch; | ||||
|     } | ||||
|  | ||||
| #ifdef AT_PRINT_RAW_CMD | ||||
|     at_print_raw_cmd("recvline", client->recv_line_buf, client->recv_line_len); | ||||
| #endif | ||||
|  | ||||
|     return client->recv_line_len; | ||||
| } | ||||
|  | ||||
| static void client_parser(at_client_t client) | ||||
| { | ||||
|     while(1) | ||||
|     { | ||||
|         if (at_recv_readline(client) > 0) | ||||
|         { | ||||
|             if (client->urc != RT_NULL) | ||||
|             { | ||||
|                 /* current receive is request, try to execute related operations */ | ||||
|                 if (client->urc->func != RT_NULL) | ||||
|                 { | ||||
|                     client->urc->func(client, client->recv_line_buf, client->recv_line_len); | ||||
|                 } | ||||
|                 client->urc = RT_NULL; | ||||
|             } | ||||
|             else if (client->resp != RT_NULL) | ||||
|             { | ||||
|                 at_response_t resp = client->resp; | ||||
|  | ||||
|                 char end_ch = client->recv_line_buf[client->recv_line_len - 1]; | ||||
|  | ||||
|                 /* current receive is response */ | ||||
|                 client->recv_line_buf[client->recv_line_len - 1] = '\0'; | ||||
|                 if (resp->buf_len + client->recv_line_len < resp->buf_size) | ||||
|                 { | ||||
|                     /* copy response lines, separated by '\0' */ | ||||
|                     rt_memcpy(resp->buf + resp->buf_len, client->recv_line_buf, client->recv_line_len); | ||||
|  | ||||
|                     /* update the current response information */ | ||||
|                     resp->buf_len += client->recv_line_len; | ||||
|                     resp->line_counts++; | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     client->resp_status = AT_RESP_BUFF_FULL; | ||||
|                     LOG_E("Read response buffer failed. The Response buffer size is out of buffer size(%d)!", resp->buf_size); | ||||
|                 } | ||||
|                 /* check response result */ | ||||
|                 if ((client->end_sign != 0) && (end_ch == client->end_sign) && (resp->line_num == 0)) | ||||
|                 { | ||||
|                     /* get the end sign, return response state END_OK.*/ | ||||
|                     client->resp_status = AT_RESP_OK; | ||||
|                 } | ||||
|                 else if (rt_memcmp(client->recv_line_buf, AT_RESP_END_OK, rt_strlen(AT_RESP_END_OK)) == 0 | ||||
|                         && resp->line_num == 0) | ||||
|                 { | ||||
|                     /* get the end data by response result, return response state END_OK. */ | ||||
|                     client->resp_status = AT_RESP_OK; | ||||
|                 } | ||||
|                 else if (rt_strstr(client->recv_line_buf, AT_RESP_END_ERROR) | ||||
|                         || (rt_memcmp(client->recv_line_buf, AT_RESP_END_FAIL, rt_strlen(AT_RESP_END_FAIL)) == 0)) | ||||
|                 { | ||||
|                     client->resp_status = AT_RESP_ERROR; | ||||
|                 } | ||||
|                 else if (resp->line_counts == resp->line_num && resp->line_num) | ||||
|                 { | ||||
|                     /* get the end data by response line, return response state END_OK.*/ | ||||
|                     client->resp_status = AT_RESP_OK; | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     continue; | ||||
|                 } | ||||
|  | ||||
|                 client->resp = RT_NULL; | ||||
|                 rt_sem_release(client->resp_notice); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
| //                log_d("unrecognized line: %.*s", client->recv_line_len, client->recv_line_buf); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| static rt_err_t at_client_rx_ind(rt_device_t dev, rt_size_t size) | ||||
| { | ||||
|     int idx = 0; | ||||
|  | ||||
|     for (idx = 0; idx < AT_CLIENT_NUM_MAX; idx++) | ||||
|     { | ||||
|         if (at_client_table[idx].device == dev && size > 0) | ||||
|         { | ||||
|             rt_sem_release(at_client_table[idx].rx_notice); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| /* initialize the client object parameters */ | ||||
| static int at_client_para_init(at_client_t client) | ||||
| { | ||||
| #define AT_CLIENT_LOCK_NAME            "at_c" | ||||
| #define AT_CLIENT_SEM_NAME             "at_cs" | ||||
| #define AT_CLIENT_RESP_NAME            "at_cr" | ||||
| #define AT_CLIENT_THREAD_NAME          "at_clnt" | ||||
|  | ||||
|     int result = RT_EOK; | ||||
|     static int at_client_num = 0; | ||||
|     char name[RT_NAME_MAX]; | ||||
|  | ||||
|     client->status = AT_STATUS_UNINITIALIZED; | ||||
|  | ||||
|     client->recv_line_len = 0; | ||||
|     client->recv_line_buf = (char *) rt_calloc(1, client->recv_bufsz); | ||||
|     if (client->recv_line_buf == RT_NULL) | ||||
|     { | ||||
|         LOG_E("AT client initialize failed! No memory for receive buffer."); | ||||
|         result = -RT_ENOMEM; | ||||
|         goto __exit; | ||||
|     } | ||||
|  | ||||
|     client->last_cmd_len = 0; | ||||
|     client->send_buf = (char *) rt_calloc(1, client->send_bufsz); | ||||
|     if (client->send_buf == RT_NULL) | ||||
|     { | ||||
|         LOG_E("AT client initialize failed! No memory for send buffer."); | ||||
|         result = -RT_ENOMEM; | ||||
|         goto __exit; | ||||
|     } | ||||
|  | ||||
|     rt_snprintf(name, RT_NAME_MAX, "%s%d", AT_CLIENT_LOCK_NAME, at_client_num); | ||||
|     client->lock = rt_mutex_create(name, RT_IPC_FLAG_PRIO); | ||||
|     if (client->lock == RT_NULL) | ||||
|     { | ||||
|         LOG_E("AT client initialize failed! at_client_recv_lock create failed!"); | ||||
|         result = -RT_ENOMEM; | ||||
|         goto __exit; | ||||
|     } | ||||
|  | ||||
|     rt_snprintf(name, RT_NAME_MAX, "%s%d", AT_CLIENT_SEM_NAME, at_client_num); | ||||
|     client->rx_notice = rt_sem_create(name, 0, RT_IPC_FLAG_FIFO); | ||||
|     if (client->rx_notice == RT_NULL) | ||||
|     { | ||||
|         LOG_E("AT client initialize failed! at_client_notice semaphore create failed!"); | ||||
|         result = -RT_ENOMEM; | ||||
|         goto __exit; | ||||
|     } | ||||
|  | ||||
|     rt_snprintf(name, RT_NAME_MAX, "%s%d", AT_CLIENT_RESP_NAME, at_client_num); | ||||
|     client->resp_notice = rt_sem_create(name, 0, RT_IPC_FLAG_FIFO); | ||||
|     if (client->resp_notice == RT_NULL) | ||||
|     { | ||||
|         LOG_E("AT client initialize failed! at_client_resp semaphore create failed!"); | ||||
|         result = -RT_ENOMEM; | ||||
|         goto __exit; | ||||
|     } | ||||
|  | ||||
|     client->urc_table = RT_NULL; | ||||
|     client->urc_table_size = 0; | ||||
|  | ||||
|     rt_snprintf(name, RT_NAME_MAX, "%s%d", AT_CLIENT_THREAD_NAME, at_client_num); | ||||
|     client->parser = rt_thread_create(name, | ||||
|                                      (void (*)(void *parameter))client_parser, | ||||
|                                      client, | ||||
|                                      1024 + 512, | ||||
|                                      RT_THREAD_PRIORITY_MAX / 3 - 1, | ||||
|                                      5); | ||||
|     if (client->parser == RT_NULL) | ||||
|     { | ||||
|         result = -RT_ENOMEM; | ||||
|     } | ||||
|  | ||||
| __exit: | ||||
|     if (result != RT_EOK) | ||||
|     { | ||||
|         if (client->lock) | ||||
|         { | ||||
|             rt_mutex_delete(client->lock); | ||||
|         } | ||||
|  | ||||
|         if (client->rx_notice) | ||||
|         { | ||||
|             rt_sem_delete(client->rx_notice); | ||||
|         } | ||||
|  | ||||
|         if (client->resp_notice) | ||||
|         { | ||||
|             rt_sem_delete(client->resp_notice); | ||||
|         } | ||||
|  | ||||
|         if (client->recv_line_buf) | ||||
|         { | ||||
|             rt_free(client->recv_line_buf); | ||||
|         } | ||||
|  | ||||
|         if (client->send_buf) | ||||
|         { | ||||
|             rt_free(client->send_buf); | ||||
|         } | ||||
|  | ||||
|         rt_memset(client, 0x00, sizeof(struct at_client)); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         at_client_num++; | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * AT client initialize. | ||||
|  * | ||||
|  * @param dev_name AT client device name | ||||
|  * @param recv_bufsz the maximum number of receive buffer length | ||||
|  * @param send_bufsz the maximum number of send command length | ||||
|  * | ||||
|  * @return 0 : initialize success | ||||
|  *        -1 : initialize failed | ||||
|  *        -5 : no memory | ||||
|  */ | ||||
| int at_client_init(const char *dev_name, rt_size_t recv_bufsz, rt_size_t send_bufsz) | ||||
| { | ||||
|     int idx = 0; | ||||
|     int result = RT_EOK; | ||||
|     rt_err_t open_result = RT_EOK; | ||||
|     at_client_t client = RT_NULL; | ||||
|  | ||||
|     RT_ASSERT(dev_name); | ||||
|     RT_ASSERT(recv_bufsz > 0); | ||||
|     RT_ASSERT(send_bufsz > 0); | ||||
|  | ||||
|     if (at_client_get(dev_name) != RT_NULL) | ||||
|     { | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     for (idx = 0; idx < AT_CLIENT_NUM_MAX && at_client_table[idx].device; idx++); | ||||
|  | ||||
|     if (idx >= AT_CLIENT_NUM_MAX) | ||||
|     { | ||||
|         LOG_E("AT client initialize failed! Check the maximum number(%d) of AT client.", AT_CLIENT_NUM_MAX); | ||||
|         result = -RT_EFULL; | ||||
|         goto __exit; | ||||
|     } | ||||
|  | ||||
|     client = &at_client_table[idx]; | ||||
|     client->recv_bufsz = recv_bufsz; | ||||
|     client->send_bufsz = send_bufsz; | ||||
|  | ||||
|     result = at_client_para_init(client); | ||||
|     if (result != RT_EOK) | ||||
|     { | ||||
|         goto __exit; | ||||
|     } | ||||
|  | ||||
|     /* find and open command device */ | ||||
|     client->device = rt_device_find(dev_name); | ||||
|     if (client->device) | ||||
|     { | ||||
|         RT_ASSERT(client->device->type == RT_Device_Class_Char); | ||||
|  | ||||
|         rt_device_set_rx_indicate(client->device, at_client_rx_ind); | ||||
|         /* using DMA mode first */ | ||||
|         open_result = rt_device_open(client->device, RT_DEVICE_OFLAG_RDWR | RT_DEVICE_FLAG_DMA_RX); | ||||
|         /* using interrupt mode when DMA mode not supported */ | ||||
|         if (open_result == -RT_EIO) | ||||
|         { | ||||
|             open_result = rt_device_open(client->device, RT_DEVICE_OFLAG_RDWR | RT_DEVICE_FLAG_INT_RX); | ||||
|         } | ||||
|         RT_ASSERT(open_result == RT_EOK); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         LOG_E("AT client initialize failed! Not find the device(%s).", dev_name); | ||||
|         result = -RT_ERROR; | ||||
|     } | ||||
|  | ||||
| __exit: | ||||
|     if (result == RT_EOK) | ||||
|     { | ||||
|         client->status = AT_STATUS_INITIALIZED; | ||||
|  | ||||
|         rt_thread_startup(client->parser); | ||||
|  | ||||
|         LOG_I("AT client(V%s) on device %s initialize success.", AT_SW_VERSION, dev_name); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         LOG_E("AT client(V%s) on device %s initialize failed(%d).", AT_SW_VERSION, dev_name, result); | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
| #endif /* AT_USING_CLIENT */ | ||||
							
								
								
									
										640
									
								
								riscv/rtthread/components/net/at/src/at_server.c
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										640
									
								
								riscv/rtthread/components/net/at/src/at_server.c
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,640 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2018-03-30     chenyong     first version | ||||
|  * 2018-04-14     chenyong     modify parse arguments | ||||
|  */ | ||||
|  | ||||
| #include <at.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
| #include <string.h> | ||||
|  | ||||
| #include <rthw.h> | ||||
|  | ||||
| #define LOG_TAG              "at.svr" | ||||
| #include <at_log.h> | ||||
|  | ||||
| #ifdef AT_USING_SERVER | ||||
|  | ||||
| #define AT_CMD_CHAR_0                  '0' | ||||
| #define AT_CMD_CHAR_9                  '9' | ||||
| #define AT_CMD_QUESTION_MARK           '?' | ||||
| #define AT_CMD_EQUAL_MARK              '=' | ||||
| #define AT_CMD_L_SQ_BRACKET            '[' | ||||
| #define AT_CMD_R_SQ_BRACKET            ']' | ||||
| #define AT_CMD_L_ANGLE_BRACKET         '<' | ||||
| #define AT_CMD_R_ANGLE_BRACKET         '>' | ||||
| #define AT_CMD_COMMA_MARK              ',' | ||||
| #define AT_CMD_SEMICOLON               ';' | ||||
| #define AT_CMD_CR                      '\r' | ||||
| #define AT_CMD_LF                      '\n' | ||||
| #define AT_CMD_NULL                    '\0' | ||||
|  | ||||
| static at_server_t at_server_local = RT_NULL; | ||||
| static at_cmd_t cmd_table = RT_NULL; | ||||
| static rt_size_t cmd_num; | ||||
|  | ||||
| extern rt_size_t at_utils_send(rt_device_t dev, | ||||
|                                rt_off_t    pos, | ||||
|                                const void *buffer, | ||||
|                                rt_size_t   size); | ||||
| extern void at_vprintf(rt_device_t device, char *send_buf, rt_size_t buf_size, const char *format, va_list args); | ||||
| extern void at_vprintfln(rt_device_t device, char *send_buf, rt_size_t buf_size, const char *format, va_list args); | ||||
|  | ||||
| /** | ||||
|  * AT server send data to AT device | ||||
|  * | ||||
|  * @param format the input format | ||||
|  */ | ||||
| void at_server_printf(const char *format, ...) | ||||
| { | ||||
|     va_list args; | ||||
|  | ||||
|     va_start(args, format); | ||||
|  | ||||
|     at_vprintf(at_server_local->device, at_server_local->send_buffer, sizeof(at_server_local->send_buffer), format, args); | ||||
|  | ||||
|     va_end(args); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * AT server send data and newline to AT device | ||||
|  * | ||||
|  * @param format the input format | ||||
|  */ | ||||
| void at_server_printfln(const char *format, ...) | ||||
| { | ||||
|     va_list args; | ||||
|  | ||||
|     va_start(args, format); | ||||
|  | ||||
|     at_vprintfln(at_server_local->device, at_server_local->send_buffer, sizeof(at_server_local->send_buffer), format, args); | ||||
|  | ||||
|     va_end(args); | ||||
| } | ||||
|  | ||||
|  | ||||
| /** | ||||
|  * AT server request arguments parse arguments | ||||
|  * | ||||
|  * @param req_args request arguments | ||||
|  * @param req_expr request expression | ||||
|  * | ||||
|  * @return  -1 : parse arguments failed | ||||
|  *           0 : parse without match | ||||
|  *          >0 : The number of arguments successfully parsed | ||||
|  */ | ||||
| int at_req_parse_args(const char *req_args, const char *req_expr, ...) | ||||
| { | ||||
|     va_list args; | ||||
|     int req_args_num = 0; | ||||
|  | ||||
|     RT_ASSERT(req_args); | ||||
|     RT_ASSERT(req_expr); | ||||
|  | ||||
|     va_start(args, req_expr); | ||||
|  | ||||
|     req_args_num = vsscanf(req_args, req_expr, args); | ||||
|  | ||||
|     va_end(args); | ||||
|  | ||||
|     return req_args_num; | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * AT server send command execute result to AT device | ||||
|  * | ||||
|  * @param result AT command execute result | ||||
|  */ | ||||
| void at_server_print_result(at_result_t result) | ||||
| { | ||||
|     switch (result) | ||||
|     { | ||||
|     case AT_RESULT_OK: | ||||
|         at_server_printfln(""); | ||||
|         at_server_printfln("OK"); | ||||
|         break; | ||||
|  | ||||
|     case AT_RESULT_FAILE: | ||||
|         at_server_printfln(""); | ||||
|         at_server_printfln("ERROR"); | ||||
|         break; | ||||
|  | ||||
|     case AT_RESULT_NULL: | ||||
|         break; | ||||
|  | ||||
|     case AT_RESULT_CMD_ERR: | ||||
|         at_server_printfln("ERR CMD MATCH FAILED!"); | ||||
|         at_server_print_result(AT_RESULT_FAILE); | ||||
|         break; | ||||
|  | ||||
|     case AT_RESULT_CHECK_FAILE: | ||||
|         at_server_printfln("ERR CHECK ARGS FORMAT FAILED!"); | ||||
|         at_server_print_result(AT_RESULT_FAILE); | ||||
|         break; | ||||
|  | ||||
|     case AT_RESULT_PARSE_FAILE: | ||||
|         at_server_printfln("ERR PARSE ARGS FAILED!"); | ||||
|         at_server_print_result(AT_RESULT_FAILE); | ||||
|         break; | ||||
|  | ||||
|     default: | ||||
|         break; | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  *  AT server print all commands to AT device | ||||
|  */ | ||||
| void rt_at_server_print_all_cmd(void) | ||||
| { | ||||
|     rt_size_t i = 0; | ||||
|  | ||||
|     at_server_printfln("Commands list : "); | ||||
|  | ||||
|     for (i = 0; i < cmd_num; i++) | ||||
|     { | ||||
|         at_server_printf("%s", cmd_table[i].name); | ||||
|  | ||||
|         if (cmd_table[i].args_expr) | ||||
|         { | ||||
|             at_server_printfln("%s", cmd_table[i].args_expr); | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             at_server_printf("%c%c", AT_CMD_CR, AT_CMD_LF); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * Send data to AT Client by uart device. | ||||
|  * | ||||
|  * @param server current AT server object | ||||
|  * @param buf   send data buffer | ||||
|  * @param size  send fixed data size | ||||
|  * | ||||
|  * @return >0: send data size | ||||
|  *         =0: send failed | ||||
|  */ | ||||
| rt_size_t at_server_send(at_server_t server, const char *buf, rt_size_t size) | ||||
| { | ||||
|     RT_ASSERT(buf); | ||||
|  | ||||
|     if (server == RT_NULL) | ||||
|     { | ||||
|         LOG_E("input AT Server object is NULL, please create or get AT Server object!"); | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     return at_utils_send(server->device, 0, buf, size); | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * AT Server receive fixed-length data. | ||||
|  * | ||||
|  * @param client current AT Server object | ||||
|  * @param buf   receive data buffer | ||||
|  * @param size  receive fixed data size | ||||
|  * @param timeout  receive data timeout (ms) | ||||
|  * | ||||
|  * @note this function can only be used in execution function of AT commands | ||||
|  * | ||||
|  * @return >0: receive data size | ||||
|  *         =0: receive failed | ||||
|  */ | ||||
| rt_size_t at_server_recv(at_server_t server, char *buf, rt_size_t size, rt_int32_t timeout) | ||||
| { | ||||
|     rt_size_t read_idx = 0; | ||||
|     rt_err_t result = RT_EOK; | ||||
|     char ch = 0; | ||||
|  | ||||
|     RT_ASSERT(buf); | ||||
|  | ||||
|     if (server == RT_NULL) | ||||
|     { | ||||
|         LOG_E("input AT Server object is NULL, please create or get AT Server object!"); | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     while (1) | ||||
|     { | ||||
|         if (read_idx < size) | ||||
|         { | ||||
|             /* check get data value */ | ||||
|             result = server->get_char(server, &ch, timeout); | ||||
|             if (result != RT_EOK) | ||||
|             { | ||||
|                 LOG_E("AT Server receive failed, uart device get data error."); | ||||
|                 return 0; | ||||
|             } | ||||
|  | ||||
|             buf[read_idx++] = ch; | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return read_idx; | ||||
| } | ||||
|  | ||||
| at_server_t at_get_server(void) | ||||
| { | ||||
|     RT_ASSERT(at_server_local); | ||||
|     RT_ASSERT(at_server_local->status != AT_STATUS_UNINITIALIZED); | ||||
|  | ||||
|     return at_server_local; | ||||
| } | ||||
|  | ||||
| static rt_err_t at_check_args(const char *args, const char *args_format) | ||||
| { | ||||
|     rt_size_t left_sq_bracket_num = 0, right_sq_bracket_num = 0; | ||||
|     rt_size_t left_angle_bracket_num = 0, right_angle_bracket_num = 0; | ||||
|     rt_size_t comma_mark_num = 0; | ||||
|     rt_size_t i = 0; | ||||
|  | ||||
|     RT_ASSERT(args); | ||||
|     RT_ASSERT(args_format); | ||||
|  | ||||
|     for (i = 0; i < strlen(args_format); i++) | ||||
|     { | ||||
|         switch (args_format[i]) | ||||
|         { | ||||
|         case AT_CMD_L_SQ_BRACKET: | ||||
|             left_sq_bracket_num++; | ||||
|             break; | ||||
|  | ||||
|         case AT_CMD_R_SQ_BRACKET: | ||||
|             right_sq_bracket_num++; | ||||
|             break; | ||||
|  | ||||
|         case AT_CMD_L_ANGLE_BRACKET: | ||||
|             left_angle_bracket_num++; | ||||
|             break; | ||||
|  | ||||
|         case AT_CMD_R_ANGLE_BRACKET: | ||||
|             right_angle_bracket_num++; | ||||
|             break; | ||||
|  | ||||
|         default: | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if (left_sq_bracket_num != right_sq_bracket_num || left_angle_bracket_num != right_angle_bracket_num | ||||
|             || left_sq_bracket_num > left_angle_bracket_num) | ||||
|     { | ||||
|         return -RT_ERROR; | ||||
|     } | ||||
|  | ||||
|     for (i = 0; i < strlen(args); i++) | ||||
|     { | ||||
|         if (args[i] == AT_CMD_COMMA_MARK) | ||||
|         { | ||||
|             comma_mark_num++; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     if ((comma_mark_num + 1 < left_angle_bracket_num - left_sq_bracket_num) | ||||
|             || comma_mark_num + 1 > left_angle_bracket_num) | ||||
|     { | ||||
|         return -RT_ERROR; | ||||
|     } | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| static at_result_t at_cmd_process(at_cmd_t cmd, const char *cmd_args) | ||||
| { | ||||
|     RT_ASSERT(cmd); | ||||
|     RT_ASSERT(cmd_args); | ||||
|  | ||||
|     /* AT+TEST=? */ | ||||
|     if (cmd_args[0] == AT_CMD_EQUAL_MARK && cmd_args[1] == AT_CMD_QUESTION_MARK) | ||||
|     { | ||||
|         if (cmd->test == RT_NULL) | ||||
|         { | ||||
|             return AT_RESULT_CMD_ERR; | ||||
|         } | ||||
|  | ||||
|         return cmd->test(); | ||||
|     } | ||||
|     /* AT+TEST? */ | ||||
|     else if (cmd_args[0] == AT_CMD_QUESTION_MARK) | ||||
|     { | ||||
|         if (cmd->query == RT_NULL) | ||||
|         { | ||||
|             return AT_RESULT_CMD_ERR; | ||||
|         } | ||||
|  | ||||
|         return cmd->query(); | ||||
|     } | ||||
|     /* AT+TEST=1 or ATE1 */ | ||||
|     else if (cmd_args[0] == AT_CMD_EQUAL_MARK | ||||
|             || (cmd_args[0] >= AT_CMD_CHAR_0 && cmd_args[0] <= AT_CMD_CHAR_9 && (cmd_args[1] == AT_CMD_CR || cmd_args[1] == AT_CMD_LF))) | ||||
|     { | ||||
|         if (cmd->setup == RT_NULL) | ||||
|         { | ||||
|             return AT_RESULT_CMD_ERR; | ||||
|         } | ||||
|  | ||||
|         if(at_check_args(cmd_args, cmd->args_expr) < 0) | ||||
|         { | ||||
|             return AT_RESULT_CHECK_FAILE; | ||||
|         } | ||||
|  | ||||
|         return cmd->setup(cmd_args); | ||||
|     } | ||||
|     /* AT+TEST */ | ||||
|     else if (cmd_args[0] == AT_CMD_CR || cmd_args[0] == AT_CMD_LF) | ||||
|     { | ||||
|         if (cmd->exec == RT_NULL) | ||||
|         { | ||||
|             return AT_RESULT_CMD_ERR; | ||||
|         } | ||||
|  | ||||
|         return cmd->exec(); | ||||
|     } | ||||
|  | ||||
|     return AT_RESULT_FAILE; | ||||
| } | ||||
|  | ||||
| static at_cmd_t at_find_cmd(const char *cmd) | ||||
| { | ||||
|     rt_size_t i = 0; | ||||
|  | ||||
|     RT_ASSERT(cmd_table); | ||||
|  | ||||
|     for (i = 0; i < cmd_num; i++) | ||||
|     { | ||||
|         if (!strcasecmp(cmd, cmd_table[i].name)) | ||||
|         { | ||||
|             return &cmd_table[i]; | ||||
|         } | ||||
|     } | ||||
|     return RT_NULL; | ||||
| } | ||||
|  | ||||
| static rt_err_t at_cmd_get_name(const char *cmd_buffer, char *cmd_name) | ||||
| { | ||||
|     rt_size_t cmd_name_len = 0, i = 0; | ||||
|  | ||||
|     RT_ASSERT(cmd_name); | ||||
|     RT_ASSERT(cmd_buffer); | ||||
|  | ||||
|     for (i = 0; i < strlen(cmd_buffer) && i < AT_CMD_NAME_LEN; i++) | ||||
|     { | ||||
|         if (*(cmd_buffer + i) == AT_CMD_QUESTION_MARK || *(cmd_buffer + i) == AT_CMD_EQUAL_MARK | ||||
|                 || *(cmd_buffer + i) == AT_CMD_CR || *(cmd_buffer + i) == AT_CMD_LF | ||||
|                 || (*(cmd_buffer + i) >= AT_CMD_CHAR_0 && *(cmd_buffer + i) <= AT_CMD_CHAR_9)) | ||||
|         { | ||||
|             cmd_name_len = i; | ||||
|             rt_memcpy(cmd_name, cmd_buffer, cmd_name_len); | ||||
|             *(cmd_name + cmd_name_len) = '\0'; | ||||
|  | ||||
|             return RT_EOK; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return -RT_ERROR; | ||||
| } | ||||
|  | ||||
| static rt_err_t at_server_getchar(at_server_t server, char *ch, rt_int32_t timeout) | ||||
| { | ||||
|     rt_err_t result = RT_EOK; | ||||
|  | ||||
|     while (rt_device_read(at_server_local->device, 0, ch, 1) == 0) | ||||
|     { | ||||
|         rt_sem_control(at_server_local->rx_notice, RT_IPC_CMD_RESET, RT_NULL); | ||||
|         result = rt_sem_take(at_server_local->rx_notice, rt_tick_from_millisecond(timeout)); | ||||
|         if (result != RT_EOK) | ||||
|         { | ||||
|             return result; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
|  | ||||
| static void server_parser(at_server_t server) | ||||
| { | ||||
| #define ESC_KEY                 0x1B | ||||
| #define BACKSPACE_KEY           0x08 | ||||
| #define DELECT_KEY              0x7F | ||||
|  | ||||
|     char cur_cmd_name[AT_CMD_NAME_LEN] = { 0 }; | ||||
|     at_cmd_t cur_cmd = RT_NULL; | ||||
|     char *cur_cmd_args = RT_NULL, ch, last_ch; | ||||
|     at_result_t result; | ||||
|  | ||||
|     RT_ASSERT(server); | ||||
|     RT_ASSERT(server->status != AT_STATUS_UNINITIALIZED); | ||||
|  | ||||
|     while (1) | ||||
|     { | ||||
|         server->get_char(server, &ch, RT_WAITING_FOREVER); | ||||
|         if (ESC_KEY == ch) | ||||
|         { | ||||
|             break; | ||||
|         } | ||||
|  | ||||
|         if (ch == BACKSPACE_KEY || ch == DELECT_KEY) | ||||
|         { | ||||
|             if (server->cur_recv_len) | ||||
|                 server->cur_recv_len--; | ||||
|             if (server->echo_mode) | ||||
|                 at_server_printf("\b \b"); | ||||
|         } | ||||
|         else if (ch == AT_CMD_LF && last_ch == AT_CMD_CR) | ||||
|         { | ||||
|             /* skip '\n' if we get "\r\n" */ | ||||
|         } | ||||
|         else | ||||
|         { | ||||
|             if (server->cur_recv_len < sizeof(server->recv_buffer) - 1) | ||||
|             { | ||||
|                 server->recv_buffer[server->cur_recv_len++] = ch; | ||||
|                 if (ch == AT_CMD_CR || ch == AT_CMD_LF) | ||||
|                 { | ||||
|                     if (server->echo_mode) | ||||
|                         at_server_printf("%c%c", AT_CMD_CR, AT_CMD_LF); | ||||
|                     server->recv_buffer[server->cur_recv_len] = '\0'; | ||||
|                     result = AT_RESULT_FAILE; | ||||
|                     if (at_cmd_get_name(server->recv_buffer, cur_cmd_name) == RT_EOK) | ||||
|                     { | ||||
|                         cur_cmd = at_find_cmd(cur_cmd_name); | ||||
|                         if (cur_cmd) | ||||
|                         { | ||||
|                             cur_cmd_args = server->recv_buffer + strlen(cur_cmd_name); | ||||
|                             result = at_cmd_process(cur_cmd, cur_cmd_args); | ||||
|                         } | ||||
|                     } | ||||
|  | ||||
|                     at_server_print_result(result); | ||||
|                     server->cur_recv_len = 0; | ||||
|                 } | ||||
|                 else | ||||
|                 { | ||||
|                     if (server->echo_mode) | ||||
|                         at_server_printf("%c", ch); | ||||
|                 } | ||||
|                 last_ch = ch; | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 /* server receive buffer overflow!!! */ | ||||
|                 server->cur_recv_len = 0; | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| static rt_err_t at_rx_ind(rt_device_t dev, rt_size_t size) | ||||
| { | ||||
|     if (size > 0) | ||||
|     { | ||||
|         rt_sem_release(at_server_local->rx_notice); | ||||
|     } | ||||
|  | ||||
|     return RT_EOK; | ||||
| } | ||||
|  | ||||
| #if defined(__ICCARM__) || defined(__ICCRX__)               /* for IAR compiler */ | ||||
| #pragma section="RtAtCmdTab" | ||||
| #endif | ||||
|  | ||||
| int at_server_init(void) | ||||
| { | ||||
|     rt_err_t result = RT_EOK; | ||||
|     rt_err_t open_result = RT_EOK; | ||||
|  | ||||
|     if (at_server_local) | ||||
|     { | ||||
|         return result; | ||||
|     } | ||||
|  | ||||
|     /* initialize the AT commands table.*/ | ||||
| #if defined(__ARMCC_VERSION)                                 /* ARM C Compiler */ | ||||
|     extern const int RtAtCmdTab$$Base; | ||||
|     extern const int RtAtCmdTab$$Limit; | ||||
|     cmd_table = (at_cmd_t)&RtAtCmdTab$$Base; | ||||
|     cmd_num = (at_cmd_t)&RtAtCmdTab$$Limit - cmd_table; | ||||
| #elif defined (__ICCARM__) || defined(__ICCRX__)      /* for IAR Compiler */ | ||||
|     cmd_table = (at_cmd_t)__section_begin("RtAtCmdTab"); | ||||
|     cmd_num = (at_cmd_t)__section_end("RtAtCmdTab") - cmd_table; | ||||
| #elif defined (__GNUC__)                             /* for GCC Compiler */ | ||||
|     extern const int __rtatcmdtab_start; | ||||
|     extern const int __rtatcmdtab_end; | ||||
|     cmd_table = (at_cmd_t)&__rtatcmdtab_start; | ||||
|     cmd_num = (at_cmd_t) &__rtatcmdtab_end - cmd_table; | ||||
| #endif /* defined(__CC_ARM) */ | ||||
|  | ||||
|     at_server_local = (at_server_t) rt_calloc(1, sizeof(struct at_server)); | ||||
|     if (!at_server_local) | ||||
|     { | ||||
|         result = -RT_ENOMEM; | ||||
|         LOG_E("AT server session initialize failed! No memory for at_server structure !"); | ||||
|         goto __exit; | ||||
|     } | ||||
|  | ||||
|     at_server_local->echo_mode = 1; | ||||
|     at_server_local->status = AT_STATUS_UNINITIALIZED; | ||||
|  | ||||
|     rt_memset(at_server_local->recv_buffer, 0x00, AT_SERVER_RECV_BUFF_LEN); | ||||
|     at_server_local->cur_recv_len = 0; | ||||
|  | ||||
|     at_server_local->rx_notice = rt_sem_create("at_svr", 0, RT_IPC_FLAG_FIFO); | ||||
|     if (!at_server_local->rx_notice) | ||||
|     { | ||||
|         LOG_E("AT server session initialize failed! at_rx_notice semaphore create failed!"); | ||||
|         result = -RT_ENOMEM; | ||||
|         goto __exit; | ||||
|     } | ||||
|  | ||||
|     /* Find and open command device */ | ||||
|     at_server_local->device = rt_device_find(AT_SERVER_DEVICE); | ||||
|     if (at_server_local->device) | ||||
|     { | ||||
|         RT_ASSERT(at_server_local->device->type == RT_Device_Class_Char); | ||||
|  | ||||
|         rt_device_set_rx_indicate(at_server_local->device, at_rx_ind); | ||||
|         /* using DMA mode first */ | ||||
|         open_result = rt_device_open(at_server_local->device, RT_DEVICE_OFLAG_RDWR | RT_DEVICE_FLAG_DMA_RX); | ||||
|         /* using interrupt mode when DMA mode not supported */ | ||||
|         if (open_result == -RT_EIO) | ||||
|         { | ||||
|             open_result = rt_device_open(at_server_local->device, RT_DEVICE_OFLAG_RDWR | RT_DEVICE_FLAG_INT_RX); | ||||
|         } | ||||
|         RT_ASSERT(open_result == RT_EOK); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         LOG_E("AT device initialize failed! Not find the device : %s.", AT_SERVER_DEVICE); | ||||
|         result = -RT_ERROR; | ||||
|         goto __exit; | ||||
|     } | ||||
|  | ||||
|     at_server_local->get_char = at_server_getchar; | ||||
|  | ||||
|     at_server_local->parser_entry = server_parser; | ||||
|     at_server_local->parser = rt_thread_create("at_svr", | ||||
|                                          (void (*)(void *parameter))server_parser, | ||||
|                                          at_server_local, | ||||
|                                          2 * 1024, | ||||
|                                          RT_THREAD_PRIORITY_MAX / 3 - 1, | ||||
|                                          5); | ||||
|     if (at_server_local->parser == RT_NULL) | ||||
|     { | ||||
|         result = -RT_ENOMEM; | ||||
|     } | ||||
|  | ||||
| __exit: | ||||
|     if (!result) | ||||
|     { | ||||
|         at_server_local->status = AT_STATUS_INITIALIZED; | ||||
|  | ||||
|         rt_thread_startup(at_server_local->parser); | ||||
|  | ||||
|         LOG_I("RT-Thread AT server (V%s) initialize success.", AT_SW_VERSION); | ||||
|     } | ||||
|     else | ||||
|     { | ||||
|         if (at_server_local) | ||||
|         { | ||||
|             if (at_server_local->rx_notice) | ||||
|             { | ||||
|                 rt_sem_delete(at_server_local->rx_notice); | ||||
|             } | ||||
|             if (at_server_local->device) | ||||
|             { | ||||
|                 rt_device_close(at_server_local->device); | ||||
|             } | ||||
|             rt_free(at_server_local); | ||||
|             at_server_local = RT_NULL; | ||||
|         } | ||||
|  | ||||
|         LOG_E("RT-Thread AT server (V%s) initialize failed(%d).", AT_SW_VERSION, result); | ||||
|     } | ||||
|  | ||||
|     return result; | ||||
| } | ||||
| INIT_COMPONENT_EXPORT(at_server_init); | ||||
|  | ||||
| rt_weak void at_port_reset(void) | ||||
| { | ||||
|     LOG_E("The reset for AT server is not implement."); | ||||
| } | ||||
|  | ||||
| rt_weak void at_port_factory_reset(void) | ||||
| { | ||||
|     LOG_E("The factory reset for AT server is not implement."); | ||||
| } | ||||
|  | ||||
| #endif /* AT_USING_SERVER */ | ||||
							
								
								
									
										99
									
								
								riscv/rtthread/components/net/at/src/at_utils.c
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										99
									
								
								riscv/rtthread/components/net/at/src/at_utils.c
									
									
									
									
									
										Executable file
									
								
							| @@ -0,0 +1,99 @@ | ||||
| /* | ||||
|  * Copyright (c) 2006-2021, RT-Thread Development Team | ||||
|  * | ||||
|  * SPDX-License-Identifier: Apache-2.0 | ||||
|  * | ||||
|  * Change Logs: | ||||
|  * Date           Author       Notes | ||||
|  * 2018-04-14     chenyong     first version | ||||
|  * 2023-06-09     CX           optimize at_vprintfln interface | ||||
|  */ | ||||
|  | ||||
| #include <at.h> | ||||
| #include <stdlib.h> | ||||
| #include <stdio.h> | ||||
|  | ||||
| /** | ||||
|  * dump hex format data to console device | ||||
|  * | ||||
|  * @param name name for hex object, it will show on log header | ||||
|  * @param buf hex buffer | ||||
|  * @param size buffer size | ||||
|  */ | ||||
| void at_print_raw_cmd(const char *name, const char *buf, rt_size_t size) | ||||
| { | ||||
| #define __is_print(ch)       ((unsigned int)((ch) - ' ') < 127u - ' ') | ||||
| #define WIDTH_SIZE           32 | ||||
|  | ||||
|     rt_size_t i, j; | ||||
|  | ||||
|     for (i = 0; i < size; i += WIDTH_SIZE) | ||||
|     { | ||||
|         rt_kprintf("[D/AT] %s: %04X-%04X: ", name, i, i + WIDTH_SIZE); | ||||
|         for (j = 0; j < WIDTH_SIZE; j++) | ||||
|         { | ||||
|             if (i + j < size) | ||||
|             { | ||||
|                 rt_kprintf("%02X ", (unsigned char)buf[i + j]); | ||||
|             } | ||||
|             else | ||||
|             { | ||||
|                 rt_kprintf("   "); | ||||
|             } | ||||
|             if ((j + 1) % 8 == 0) | ||||
|             { | ||||
|                 rt_kprintf(" "); | ||||
|             } | ||||
|         } | ||||
|         rt_kprintf("  "); | ||||
|         for (j = 0; j < WIDTH_SIZE; j++) | ||||
|         { | ||||
|             if (i + j < size) | ||||
|             { | ||||
|                 rt_kprintf("%c", __is_print(buf[i + j]) ? buf[i + j] : '.'); | ||||
|             } | ||||
|         } | ||||
|         rt_kprintf("\n"); | ||||
|     } | ||||
| } | ||||
|  | ||||
| rt_weak rt_size_t at_utils_send(rt_device_t dev, | ||||
|                                 rt_off_t    pos, | ||||
|                                 const void *buffer, | ||||
|                                 rt_size_t   size) | ||||
| { | ||||
|     return rt_device_write(dev, pos, buffer, size); | ||||
| } | ||||
|  | ||||
| rt_size_t at_vprintf(rt_device_t device, char *send_buf, rt_size_t buf_size, const char *format, va_list args) | ||||
| { | ||||
|     rt_size_t len = vsnprintf(send_buf, buf_size, format, args); | ||||
|     if (len == 0) | ||||
|     { | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
| #ifdef AT_PRINT_RAW_CMD | ||||
|     at_print_raw_cmd("sendline", send_buf, len); | ||||
| #endif | ||||
|  | ||||
|     return at_utils_send(device, 0, send_buf, len); | ||||
| } | ||||
|  | ||||
| rt_size_t at_vprintfln(rt_device_t device, char *send_buf, rt_size_t buf_size, const char *format, va_list args) | ||||
| { | ||||
|     rt_size_t len = vsnprintf(send_buf, buf_size - 2, format, args); | ||||
|     if (len == 0) | ||||
|     { | ||||
|         return 0; | ||||
|     } | ||||
|  | ||||
|     send_buf[len++] = '\r'; | ||||
|     send_buf[len++] = '\n'; | ||||
|  | ||||
| #ifdef AT_PRINT_RAW_CMD | ||||
|     at_print_raw_cmd("sendline", send_buf, len); | ||||
| #endif | ||||
|  | ||||
|     return at_utils_send(device, 0, send_buf, len); | ||||
| } | ||||
		Reference in New Issue
	
	Block a user