2023-04-10 18:24:56 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2023 Bjarki Arge Andreasen
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <zephyr/kernel.h>
|
|
|
|
#include <zephyr/device.h>
|
|
|
|
#include <zephyr/drivers/gpio.h>
|
|
|
|
#include <zephyr/modem/chat.h>
|
|
|
|
#include <zephyr/modem/cmux.h>
|
|
|
|
#include <zephyr/modem/pipe.h>
|
|
|
|
#include <zephyr/modem/ppp.h>
|
|
|
|
#include <zephyr/modem/backend/uart.h>
|
|
|
|
#include <zephyr/net/ppp.h>
|
|
|
|
#include <zephyr/pm/device.h>
|
|
|
|
#include <zephyr/sys/atomic.h>
|
|
|
|
|
|
|
|
#include <zephyr/logging/log.h>
|
|
|
|
LOG_MODULE_REGISTER(modem_cellular, CONFIG_MODEM_LOG_LEVEL);
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
2023-10-31 18:15:48 +01:00
|
|
|
#define MODEM_CELLULAR_PERIODIC_SCRIPT_TIMEOUT \
|
|
|
|
K_MSEC(CONFIG_MODEM_CELLULAR_PERIODIC_SCRIPT_MS)
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
enum modem_cellular_state {
|
|
|
|
MODEM_CELLULAR_STATE_IDLE = 0,
|
|
|
|
MODEM_CELLULAR_STATE_RESET_PULSE,
|
|
|
|
MODEM_CELLULAR_STATE_POWER_ON_PULSE,
|
|
|
|
MODEM_CELLULAR_STATE_AWAIT_POWER_ON,
|
|
|
|
MODEM_CELLULAR_STATE_RUN_INIT_SCRIPT,
|
|
|
|
MODEM_CELLULAR_STATE_CONNECT_CMUX,
|
|
|
|
MODEM_CELLULAR_STATE_OPEN_DLCI1,
|
|
|
|
MODEM_CELLULAR_STATE_OPEN_DLCI2,
|
|
|
|
MODEM_CELLULAR_STATE_RUN_DIAL_SCRIPT,
|
|
|
|
MODEM_CELLULAR_STATE_AWAIT_REGISTERED,
|
|
|
|
MODEM_CELLULAR_STATE_CARRIER_ON,
|
|
|
|
MODEM_CELLULAR_STATE_INIT_POWER_OFF,
|
|
|
|
MODEM_CELLULAR_STATE_POWER_OFF_PULSE,
|
|
|
|
MODEM_CELLULAR_STATE_AWAIT_POWER_OFF,
|
|
|
|
};
|
|
|
|
|
|
|
|
enum modem_cellular_event {
|
|
|
|
MODEM_CELLULAR_EVENT_RESUME = 0,
|
|
|
|
MODEM_CELLULAR_EVENT_SUSPEND,
|
|
|
|
MODEM_CELLULAR_EVENT_SCRIPT_SUCCESS,
|
|
|
|
MODEM_CELLULAR_EVENT_SCRIPT_FAILED,
|
|
|
|
MODEM_CELLULAR_EVENT_CMUX_CONNECTED,
|
|
|
|
MODEM_CELLULAR_EVENT_DLCI1_OPENED,
|
|
|
|
MODEM_CELLULAR_EVENT_DLCI2_OPENED,
|
|
|
|
MODEM_CELLULAR_EVENT_TIMEOUT,
|
|
|
|
MODEM_CELLULAR_EVENT_REGISTERED,
|
|
|
|
MODEM_CELLULAR_EVENT_DEREGISTERED,
|
2023-09-05 09:20:32 +02:00
|
|
|
MODEM_CELLULAR_EVENT_BUS_OPENED,
|
|
|
|
MODEM_CELLULAR_EVENT_BUS_CLOSED,
|
2023-04-10 18:24:56 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
struct modem_cellular_data {
|
|
|
|
/* UART backend */
|
|
|
|
struct modem_pipe *uart_pipe;
|
|
|
|
struct modem_backend_uart uart_backend;
|
|
|
|
uint8_t uart_backend_receive_buf[512];
|
|
|
|
uint8_t uart_backend_transmit_buf[512];
|
|
|
|
|
|
|
|
/* CMUX */
|
|
|
|
struct modem_cmux cmux;
|
|
|
|
uint8_t cmux_receive_buf[128];
|
|
|
|
uint8_t cmux_transmit_buf[256];
|
|
|
|
struct modem_cmux_dlci dlci1;
|
|
|
|
struct modem_cmux_dlci dlci2;
|
|
|
|
struct modem_pipe *dlci1_pipe;
|
|
|
|
struct modem_pipe *dlci2_pipe;
|
|
|
|
uint8_t dlci1_receive_buf[128];
|
|
|
|
uint8_t dlci2_receive_buf[256];
|
|
|
|
|
|
|
|
/* Modem chat */
|
|
|
|
struct modem_chat chat;
|
|
|
|
uint8_t chat_receive_buf[128];
|
|
|
|
uint8_t chat_delimiter[1];
|
|
|
|
uint8_t chat_filter[1];
|
|
|
|
uint8_t *chat_argv[32];
|
|
|
|
|
|
|
|
/* Status */
|
|
|
|
uint8_t imei[15];
|
|
|
|
uint8_t hwinfo[64];
|
|
|
|
uint8_t registration_status_gsm;
|
|
|
|
uint8_t registration_status_gprs;
|
|
|
|
uint8_t registration_status_lte;
|
|
|
|
|
|
|
|
/* PPP */
|
|
|
|
struct modem_ppp *ppp;
|
|
|
|
|
|
|
|
enum modem_cellular_state state;
|
|
|
|
const struct device *dev;
|
|
|
|
struct k_work_delayable timeout_work;
|
|
|
|
|
|
|
|
/* Power management */
|
|
|
|
struct k_sem suspended_sem;
|
|
|
|
|
|
|
|
/* Event dispatcher */
|
|
|
|
struct k_work event_dispatch_work;
|
|
|
|
uint8_t event_buf[8];
|
|
|
|
struct ring_buf event_rb;
|
|
|
|
struct k_mutex event_rb_lock;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct modem_cellular_config {
|
|
|
|
const struct device *uart;
|
|
|
|
const struct gpio_dt_spec power_gpio;
|
|
|
|
const struct gpio_dt_spec reset_gpio;
|
|
|
|
const uint16_t power_pulse_duration_ms;
|
|
|
|
const uint16_t reset_pulse_duration_ms;
|
|
|
|
const uint16_t startup_time_ms;
|
|
|
|
const uint16_t shutdown_time_ms;
|
|
|
|
const struct modem_chat_script *init_chat_script;
|
|
|
|
const struct modem_chat_script *dial_chat_script;
|
2023-10-31 18:15:48 +01:00
|
|
|
const struct modem_chat_script *periodic_chat_script;
|
2023-04-10 18:24:56 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static const char *modem_cellular_state_str(enum modem_cellular_state state)
|
|
|
|
{
|
|
|
|
switch (state) {
|
|
|
|
case MODEM_CELLULAR_STATE_IDLE:
|
|
|
|
return "idle";
|
|
|
|
case MODEM_CELLULAR_STATE_RESET_PULSE:
|
|
|
|
return "reset pulse";
|
|
|
|
case MODEM_CELLULAR_STATE_POWER_ON_PULSE:
|
|
|
|
return "power pulse";
|
|
|
|
case MODEM_CELLULAR_STATE_AWAIT_POWER_ON:
|
|
|
|
return "await power on";
|
|
|
|
case MODEM_CELLULAR_STATE_RUN_INIT_SCRIPT:
|
|
|
|
return "run init script";
|
|
|
|
case MODEM_CELLULAR_STATE_CONNECT_CMUX:
|
|
|
|
return "connect cmux";
|
|
|
|
case MODEM_CELLULAR_STATE_OPEN_DLCI1:
|
|
|
|
return "open dlci1";
|
|
|
|
case MODEM_CELLULAR_STATE_OPEN_DLCI2:
|
|
|
|
return "open dlci2";
|
|
|
|
case MODEM_CELLULAR_STATE_AWAIT_REGISTERED:
|
|
|
|
return "await registered";
|
|
|
|
case MODEM_CELLULAR_STATE_RUN_DIAL_SCRIPT:
|
|
|
|
return "run dial script";
|
|
|
|
case MODEM_CELLULAR_STATE_CARRIER_ON:
|
|
|
|
return "carrier on";
|
|
|
|
case MODEM_CELLULAR_STATE_INIT_POWER_OFF:
|
|
|
|
return "init power off";
|
|
|
|
case MODEM_CELLULAR_STATE_POWER_OFF_PULSE:
|
|
|
|
return "power off pulse";
|
|
|
|
case MODEM_CELLULAR_STATE_AWAIT_POWER_OFF:
|
|
|
|
return "await power off";
|
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
static const char *modem_cellular_event_str(enum modem_cellular_event event)
|
|
|
|
{
|
|
|
|
switch (event) {
|
|
|
|
case MODEM_CELLULAR_EVENT_RESUME:
|
|
|
|
return "resume";
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
return "suspend";
|
|
|
|
case MODEM_CELLULAR_EVENT_SCRIPT_SUCCESS:
|
|
|
|
return "script success";
|
|
|
|
case MODEM_CELLULAR_EVENT_SCRIPT_FAILED:
|
|
|
|
return "script failed";
|
|
|
|
case MODEM_CELLULAR_EVENT_CMUX_CONNECTED:
|
|
|
|
return "cmux connected";
|
|
|
|
case MODEM_CELLULAR_EVENT_DLCI1_OPENED:
|
|
|
|
return "dlci1 opened";
|
|
|
|
case MODEM_CELLULAR_EVENT_DLCI2_OPENED:
|
|
|
|
return "dlci2 opened";
|
|
|
|
case MODEM_CELLULAR_EVENT_TIMEOUT:
|
|
|
|
return "timeout";
|
|
|
|
case MODEM_CELLULAR_EVENT_REGISTERED:
|
|
|
|
return "registered";
|
|
|
|
case MODEM_CELLULAR_EVENT_DEREGISTERED:
|
|
|
|
return "deregistered";
|
2023-09-05 09:20:32 +02:00
|
|
|
case MODEM_CELLULAR_EVENT_BUS_OPENED:
|
|
|
|
return "bus opened";
|
|
|
|
case MODEM_CELLULAR_EVENT_BUS_CLOSED:
|
|
|
|
return "bus closed";
|
2023-04-10 18:24:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool modem_cellular_gpio_is_enabled(const struct gpio_dt_spec *gpio)
|
|
|
|
{
|
|
|
|
return gpio->port != NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_enter_state(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_state state);
|
|
|
|
|
|
|
|
static void modem_cellular_delegate_event(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt);
|
|
|
|
|
|
|
|
static void modem_cellular_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt);
|
|
|
|
|
2023-09-05 09:20:32 +02:00
|
|
|
static void modem_cellular_bus_pipe_handler(struct modem_pipe *pipe,
|
|
|
|
enum modem_pipe_event event,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
struct modem_cellular_data *data = (struct modem_cellular_data *)user_data;
|
|
|
|
|
|
|
|
switch (event) {
|
|
|
|
case MODEM_PIPE_EVENT_OPENED:
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_BUS_OPENED);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_PIPE_EVENT_CLOSED:
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_BUS_CLOSED);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
static void modem_cellular_dlci1_pipe_handler(struct modem_pipe *pipe,
|
|
|
|
enum modem_pipe_event event,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
struct modem_cellular_data *data = (struct modem_cellular_data *)user_data;
|
|
|
|
|
|
|
|
switch (event) {
|
|
|
|
case MODEM_PIPE_EVENT_OPENED:
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_DLCI1_OPENED);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_dlci2_pipe_handler(struct modem_pipe *pipe,
|
|
|
|
enum modem_pipe_event event,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
struct modem_cellular_data *data = (struct modem_cellular_data *)user_data;
|
|
|
|
|
|
|
|
switch (event) {
|
|
|
|
case MODEM_PIPE_EVENT_OPENED:
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_DLCI2_OPENED);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_chat_callback_handler(struct modem_chat *chat,
|
|
|
|
enum modem_chat_script_result result,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
struct modem_cellular_data *data = (struct modem_cellular_data *)user_data;
|
|
|
|
|
|
|
|
if (result == MODEM_CHAT_SCRIPT_RESULT_SUCCESS) {
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_SCRIPT_SUCCESS);
|
|
|
|
} else {
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_SCRIPT_FAILED);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_chat_on_imei(struct modem_chat *chat, char **argv, uint16_t argc,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
struct modem_cellular_data *data = (struct modem_cellular_data *)user_data;
|
|
|
|
|
|
|
|
if (argc != 2) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strlen(argv[1]) != 15) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (uint8_t i = 0; i < 15; i++) {
|
|
|
|
data->imei[i] = argv[1][i] - '0';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_chat_on_cgmm(struct modem_chat *chat, char **argv, uint16_t argc,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
struct modem_cellular_data *data = (struct modem_cellular_data *)user_data;
|
|
|
|
|
|
|
|
if (argc != 2) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
strncpy(data->hwinfo, argv[1], sizeof(data->hwinfo) - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool modem_cellular_is_registered(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
return (data->registration_status_gsm == 1)
|
|
|
|
|| (data->registration_status_gsm == 5)
|
|
|
|
|| (data->registration_status_gprs == 1)
|
|
|
|
|| (data->registration_status_gprs == 5)
|
|
|
|
|| (data->registration_status_lte == 1)
|
|
|
|
|| (data->registration_status_lte == 5);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_chat_on_cxreg(struct modem_chat *chat, char **argv, uint16_t argc,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
struct modem_cellular_data *data = (struct modem_cellular_data *)user_data;
|
|
|
|
uint8_t registration_status;
|
|
|
|
bool is_registered;
|
|
|
|
|
|
|
|
is_registered = modem_cellular_is_registered(data);
|
|
|
|
|
|
|
|
if (argc == 2) {
|
|
|
|
registration_status = atoi(argv[1]);
|
|
|
|
} else if (argc == 3) {
|
|
|
|
registration_status = atoi(argv[2]);
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(argv[0], "+CREG: ") == 0) {
|
|
|
|
data->registration_status_gsm = registration_status;
|
|
|
|
} else if (strcmp(argv[0], "+CGREG: ") == 0) {
|
|
|
|
data->registration_status_gprs = registration_status;
|
|
|
|
} else {
|
|
|
|
data->registration_status_lte = registration_status;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (modem_cellular_is_registered(data)) {
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_REGISTERED);
|
|
|
|
} else {
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_DEREGISTERED);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
MODEM_CHAT_MATCH_DEFINE(ok_match, "OK", "", NULL);
|
|
|
|
MODEM_CHAT_MATCHES_DEFINE(allow_match,
|
|
|
|
MODEM_CHAT_MATCH("OK", "", NULL),
|
|
|
|
MODEM_CHAT_MATCH("ERROR", "", NULL));
|
|
|
|
|
|
|
|
MODEM_CHAT_MATCH_DEFINE(imei_match, "", "", modem_cellular_chat_on_imei);
|
|
|
|
MODEM_CHAT_MATCH_DEFINE(cgmm_match, "", "", modem_cellular_chat_on_cgmm);
|
|
|
|
|
|
|
|
MODEM_CHAT_MATCHES_DEFINE(unsol_matches,
|
|
|
|
MODEM_CHAT_MATCH("+CREG: ", ",", modem_cellular_chat_on_cxreg),
|
|
|
|
MODEM_CHAT_MATCH("+CEREG: ", ",", modem_cellular_chat_on_cxreg),
|
|
|
|
MODEM_CHAT_MATCH("+CGREG: ", ",", modem_cellular_chat_on_cxreg));
|
|
|
|
|
|
|
|
MODEM_CHAT_MATCHES_DEFINE(abort_matches, MODEM_CHAT_MATCH("ERROR", "", NULL));
|
|
|
|
|
|
|
|
MODEM_CHAT_MATCHES_DEFINE(dial_abort_matches,
|
|
|
|
MODEM_CHAT_MATCH("ERROR", "", NULL),
|
|
|
|
MODEM_CHAT_MATCH("BUSY", "", NULL),
|
|
|
|
MODEM_CHAT_MATCH("NO ANSWER", "", NULL),
|
|
|
|
MODEM_CHAT_MATCH("NO CARRIER", "", NULL),
|
|
|
|
MODEM_CHAT_MATCH("NO DIALTONE", "", NULL));
|
|
|
|
|
|
|
|
static void modem_cellular_log_state_changed(enum modem_cellular_state last_state,
|
|
|
|
enum modem_cellular_state new_state)
|
|
|
|
{
|
|
|
|
LOG_INF("switch from %s to %s", modem_cellular_state_str(last_state),
|
|
|
|
modem_cellular_state_str(new_state));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_log_event(enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
LOG_INF("event %s", modem_cellular_event_str(evt));
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_start_timer(struct modem_cellular_data *data, k_timeout_t timeout)
|
|
|
|
{
|
|
|
|
k_work_schedule(&data->timeout_work, timeout);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_stop_timer(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
k_work_cancel_delayable(&data->timeout_work);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_timeout_handler(struct k_work *item)
|
|
|
|
{
|
2023-09-01 00:17:18 +02:00
|
|
|
struct k_work_delayable *dwork = k_work_delayable_from_work(item);
|
2023-04-10 18:24:56 +02:00
|
|
|
struct modem_cellular_data *data =
|
2023-09-01 00:17:18 +02:00
|
|
|
CONTAINER_OF(dwork, struct modem_cellular_data, timeout_work);
|
2023-04-10 18:24:56 +02:00
|
|
|
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_TIMEOUT);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_event_dispatch_handler(struct k_work *item)
|
|
|
|
{
|
|
|
|
struct modem_cellular_data *data =
|
|
|
|
CONTAINER_OF(item, struct modem_cellular_data, event_dispatch_work);
|
|
|
|
|
|
|
|
uint8_t events[sizeof(data->event_buf)];
|
|
|
|
uint8_t events_cnt;
|
|
|
|
|
|
|
|
k_mutex_lock(&data->event_rb_lock, K_FOREVER);
|
|
|
|
|
|
|
|
events_cnt = (uint8_t)ring_buf_get(&data->event_rb, events, sizeof(data->event_buf));
|
|
|
|
|
|
|
|
k_mutex_unlock(&data->event_rb_lock);
|
|
|
|
|
|
|
|
for (uint8_t i = 0; i < events_cnt; i++) {
|
|
|
|
modem_cellular_event_handler(data, (enum modem_cellular_event)events[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_delegate_event(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
k_mutex_lock(&data->event_rb_lock, K_FOREVER);
|
|
|
|
ring_buf_put(&data->event_rb, (uint8_t *)&evt, 1);
|
|
|
|
k_mutex_unlock(&data->event_rb_lock);
|
|
|
|
k_work_submit(&data->event_dispatch_work);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_idle_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
if (modem_cellular_gpio_is_enabled(&config->reset_gpio)) {
|
|
|
|
gpio_pin_set_dt(&config->reset_gpio, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
modem_chat_release(&data->chat);
|
|
|
|
modem_ppp_release(data->ppp);
|
|
|
|
modem_cmux_release(&data->cmux);
|
2023-09-05 09:20:32 +02:00
|
|
|
modem_pipe_close_async(data->uart_pipe);
|
2023-04-10 18:24:56 +02:00
|
|
|
k_sem_give(&data->suspended_sem);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_idle_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
switch (evt) {
|
|
|
|
case MODEM_CELLULAR_EVENT_RESUME:
|
|
|
|
if (modem_cellular_gpio_is_enabled(&config->power_gpio)) {
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_POWER_ON_PULSE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (modem_cellular_gpio_is_enabled(&config->reset_gpio)) {
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_AWAIT_POWER_ON);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_RUN_INIT_SCRIPT);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
k_sem_give(&data->suspended_sem);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_idle_state_leave(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
k_sem_take(&data->suspended_sem, K_NO_WAIT);
|
|
|
|
|
|
|
|
if (modem_cellular_gpio_is_enabled(&config->reset_gpio)) {
|
|
|
|
gpio_pin_set_dt(&config->reset_gpio, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_reset_pulse_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
gpio_pin_set_dt(&config->reset_gpio, 1);
|
|
|
|
modem_cellular_start_timer(data, K_MSEC(config->reset_pulse_duration_ms));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_reset_pulse_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
switch (evt) {
|
|
|
|
case MODEM_CELLULAR_EVENT_TIMEOUT:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_AWAIT_POWER_ON);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_IDLE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_reset_pulse_state_leave(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
gpio_pin_set_dt(&config->reset_gpio, 0);
|
|
|
|
modem_cellular_stop_timer(data);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_power_on_pulse_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
gpio_pin_set_dt(&config->power_gpio, 1);
|
|
|
|
modem_cellular_start_timer(data, K_MSEC(config->power_pulse_duration_ms));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_power_on_pulse_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
switch (evt) {
|
|
|
|
case MODEM_CELLULAR_EVENT_TIMEOUT:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_AWAIT_POWER_ON);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_IDLE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_power_on_pulse_state_leave(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
gpio_pin_set_dt(&config->power_gpio, 0);
|
|
|
|
modem_cellular_stop_timer(data);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_await_power_on_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
modem_cellular_start_timer(data, K_MSEC(config->startup_time_ms));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_await_power_on_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
switch (evt) {
|
|
|
|
case MODEM_CELLULAR_EVENT_TIMEOUT:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_RUN_INIT_SCRIPT);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_IDLE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_run_init_script_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
2023-09-05 09:20:32 +02:00
|
|
|
modem_pipe_attach(data->uart_pipe, modem_cellular_bus_pipe_handler, data);
|
|
|
|
return modem_pipe_open_async(data->uart_pipe);
|
2023-04-10 18:24:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_run_init_script_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
switch (evt) {
|
2023-09-05 09:20:32 +02:00
|
|
|
case MODEM_CELLULAR_EVENT_BUS_OPENED:
|
|
|
|
modem_chat_attach(&data->chat, data->uart_pipe);
|
2023-09-15 17:42:27 +02:00
|
|
|
modem_chat_run_script_async(&data->chat, config->init_chat_script);
|
2023-09-05 09:20:32 +02:00
|
|
|
break;
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
case MODEM_CELLULAR_EVENT_SCRIPT_SUCCESS:
|
|
|
|
net_if_set_link_addr(modem_ppp_get_iface(data->ppp), data->imei,
|
|
|
|
ARRAY_SIZE(data->imei), NET_LINK_UNKNOWN);
|
|
|
|
|
2023-09-05 09:20:32 +02:00
|
|
|
modem_chat_release(&data->chat);
|
|
|
|
modem_pipe_attach(data->uart_pipe, modem_cellular_bus_pipe_handler, data);
|
|
|
|
modem_pipe_close_async(data->uart_pipe);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_BUS_CLOSED:
|
2023-04-10 18:24:56 +02:00
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_CONNECT_CMUX);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_IDLE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SCRIPT_FAILED:
|
|
|
|
if (modem_cellular_gpio_is_enabled(&config->power_gpio)) {
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_POWER_ON_PULSE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (modem_cellular_gpio_is_enabled(&config->reset_gpio)) {
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_RESET_PULSE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_IDLE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_connect_cmux_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
2023-09-05 09:20:32 +02:00
|
|
|
/*
|
|
|
|
* Allow modem to switch bus into CMUX mode. Some modems disable UART RX while
|
|
|
|
* switching, resulting in UART RX errors as bus is no longer pulled up by modem.
|
|
|
|
*/
|
|
|
|
modem_cellular_start_timer(data, K_MSEC(100));
|
2023-04-10 18:24:56 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_connect_cmux_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
switch (evt) {
|
|
|
|
case MODEM_CELLULAR_EVENT_TIMEOUT:
|
2023-09-05 09:20:32 +02:00
|
|
|
modem_pipe_attach(data->uart_pipe, modem_cellular_bus_pipe_handler, data);
|
|
|
|
modem_pipe_open_async(data->uart_pipe);
|
2023-10-13 10:43:53 +02:00
|
|
|
break;
|
2023-09-05 09:20:32 +02:00
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_BUS_OPENED:
|
|
|
|
modem_cmux_attach(&data->cmux, data->uart_pipe);
|
2023-04-10 18:24:56 +02:00
|
|
|
modem_cmux_connect_async(&data->cmux);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_CMUX_CONNECTED:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_OPEN_DLCI1);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_INIT_POWER_OFF);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_open_dlci1_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
modem_pipe_attach(data->dlci1_pipe, modem_cellular_dlci1_pipe_handler, data);
|
|
|
|
return modem_pipe_open_async(data->dlci1_pipe);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_open_dlci1_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
switch (evt) {
|
|
|
|
case MODEM_CELLULAR_EVENT_DLCI1_OPENED:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_OPEN_DLCI2);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_INIT_POWER_OFF);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_open_dlci1_state_leave(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
modem_pipe_release(data->dlci1_pipe);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_open_dlci2_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
modem_pipe_attach(data->dlci2_pipe, modem_cellular_dlci2_pipe_handler, data);
|
|
|
|
return modem_pipe_open_async(data->dlci2_pipe);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_open_dlci2_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
switch (evt) {
|
|
|
|
case MODEM_CELLULAR_EVENT_DLCI2_OPENED:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_RUN_DIAL_SCRIPT);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_INIT_POWER_OFF);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_open_dlci2_state_leave(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
modem_pipe_release(data->dlci2_pipe);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_run_dial_script_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
2023-09-05 09:20:32 +02:00
|
|
|
/* Allow modem time to enter command mode before running dial script */
|
|
|
|
modem_cellular_start_timer(data, K_MSEC(100));
|
|
|
|
return 0;
|
2023-04-10 18:24:56 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_run_dial_script_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
2023-09-05 09:20:32 +02:00
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
switch (evt) {
|
2023-09-05 09:20:32 +02:00
|
|
|
case MODEM_CELLULAR_EVENT_TIMEOUT:
|
|
|
|
modem_chat_attach(&data->chat, data->dlci1_pipe);
|
2023-09-15 17:42:27 +02:00
|
|
|
modem_chat_run_script_async(&data->chat, config->dial_chat_script);
|
2023-09-05 09:20:32 +02:00
|
|
|
break;
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
case MODEM_CELLULAR_EVENT_SCRIPT_SUCCESS:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_AWAIT_REGISTERED);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_INIT_POWER_OFF);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_run_dial_script_state_leave(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
modem_chat_release(&data->chat);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_await_registered_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
if (modem_ppp_attach(data->ppp, data->dlci1_pipe) < 0) {
|
|
|
|
return -EAGAIN;
|
|
|
|
}
|
|
|
|
|
2023-10-31 18:15:48 +01:00
|
|
|
modem_cellular_start_timer(data, MODEM_CELLULAR_PERIODIC_SCRIPT_TIMEOUT);
|
2023-04-10 18:24:56 +02:00
|
|
|
return modem_chat_attach(&data->chat, data->dlci2_pipe);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_await_registered_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
2023-10-31 18:15:48 +01:00
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
switch (evt) {
|
2023-10-31 18:15:48 +01:00
|
|
|
case MODEM_CELLULAR_EVENT_SCRIPT_SUCCESS:
|
|
|
|
case MODEM_CELLULAR_EVENT_SCRIPT_FAILED:
|
|
|
|
modem_cellular_start_timer(data, MODEM_CELLULAR_PERIODIC_SCRIPT_TIMEOUT);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_TIMEOUT:
|
|
|
|
modem_chat_run_script_async(&data->chat, config->periodic_chat_script);
|
|
|
|
break;
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
case MODEM_CELLULAR_EVENT_REGISTERED:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_CARRIER_ON);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_INIT_POWER_OFF);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-31 18:15:48 +01:00
|
|
|
static int modem_cellular_on_await_registered_state_leave(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
modem_cellular_stop_timer(data);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
static int modem_cellular_on_carrier_on_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
net_if_carrier_on(modem_ppp_get_iface(data->ppp));
|
2023-10-31 18:15:48 +01:00
|
|
|
modem_cellular_start_timer(data, MODEM_CELLULAR_PERIODIC_SCRIPT_TIMEOUT);
|
2023-04-10 18:24:56 +02:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_carrier_on_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
2023-10-31 18:15:48 +01:00
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
switch (evt) {
|
2023-10-31 18:15:48 +01:00
|
|
|
case MODEM_CELLULAR_EVENT_SCRIPT_SUCCESS:
|
|
|
|
case MODEM_CELLULAR_EVENT_SCRIPT_FAILED:
|
|
|
|
modem_cellular_start_timer(data, MODEM_CELLULAR_PERIODIC_SCRIPT_TIMEOUT);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_TIMEOUT:
|
|
|
|
modem_chat_run_script_async(&data->chat, config->periodic_chat_script);
|
|
|
|
break;
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
case MODEM_CELLULAR_EVENT_DEREGISTERED:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_RUN_DIAL_SCRIPT);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_EVENT_SUSPEND:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_INIT_POWER_OFF);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_carrier_on_state_leave(struct modem_cellular_data *data)
|
|
|
|
{
|
2023-10-31 18:15:48 +01:00
|
|
|
modem_cellular_stop_timer(data);
|
2023-04-10 18:24:56 +02:00
|
|
|
net_if_carrier_off(modem_ppp_get_iface(data->ppp));
|
|
|
|
modem_chat_release(&data->chat);
|
|
|
|
modem_ppp_release(data->ppp);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_init_power_off_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
2023-09-05 09:20:32 +02:00
|
|
|
modem_pipe_close_async(data->uart_pipe);
|
2023-04-10 18:24:56 +02:00
|
|
|
modem_cellular_start_timer(data, K_MSEC(2000));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_init_power_off_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
switch (evt) {
|
|
|
|
case MODEM_CELLULAR_EVENT_TIMEOUT:
|
|
|
|
if (modem_cellular_gpio_is_enabled(&config->power_gpio)) {
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_POWER_OFF_PULSE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_IDLE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_init_power_off_state_leave(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
modem_chat_release(&data->chat);
|
|
|
|
modem_ppp_release(data->ppp);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_power_off_pulse_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
gpio_pin_set_dt(&config->power_gpio, 1);
|
|
|
|
modem_cellular_start_timer(data, K_MSEC(config->power_pulse_duration_ms));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_power_off_pulse_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
switch (evt) {
|
|
|
|
case MODEM_CELLULAR_EVENT_TIMEOUT:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_AWAIT_POWER_OFF);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_power_off_pulse_state_leave(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
gpio_pin_set_dt(&config->power_gpio, 0);
|
|
|
|
modem_cellular_stop_timer(data);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_await_power_off_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
const struct modem_cellular_config *config =
|
|
|
|
(const struct modem_cellular_config *)data->dev->config;
|
|
|
|
|
|
|
|
modem_cellular_start_timer(data, K_MSEC(config->shutdown_time_ms));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_await_power_off_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
switch (evt) {
|
|
|
|
case MODEM_CELLULAR_EVENT_TIMEOUT:
|
|
|
|
modem_cellular_enter_state(data, MODEM_CELLULAR_STATE_IDLE);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_state_enter(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
switch (data->state) {
|
|
|
|
case MODEM_CELLULAR_STATE_IDLE:
|
|
|
|
ret = modem_cellular_on_idle_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_RESET_PULSE:
|
|
|
|
ret = modem_cellular_on_reset_pulse_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_POWER_ON_PULSE:
|
|
|
|
ret = modem_cellular_on_power_on_pulse_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_AWAIT_POWER_ON:
|
|
|
|
ret = modem_cellular_on_await_power_on_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_RUN_INIT_SCRIPT:
|
|
|
|
ret = modem_cellular_on_run_init_script_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_CONNECT_CMUX:
|
|
|
|
ret = modem_cellular_on_connect_cmux_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_OPEN_DLCI1:
|
|
|
|
ret = modem_cellular_on_open_dlci1_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_OPEN_DLCI2:
|
|
|
|
ret = modem_cellular_on_open_dlci2_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_RUN_DIAL_SCRIPT:
|
|
|
|
ret = modem_cellular_on_run_dial_script_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_AWAIT_REGISTERED:
|
|
|
|
ret = modem_cellular_on_await_registered_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_CARRIER_ON:
|
|
|
|
ret = modem_cellular_on_carrier_on_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_INIT_POWER_OFF:
|
|
|
|
ret = modem_cellular_on_init_power_off_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_POWER_OFF_PULSE:
|
|
|
|
ret = modem_cellular_on_power_off_pulse_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_AWAIT_POWER_OFF:
|
|
|
|
ret = modem_cellular_on_await_power_off_state_enter(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int modem_cellular_on_state_leave(struct modem_cellular_data *data)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
switch (data->state) {
|
|
|
|
case MODEM_CELLULAR_STATE_IDLE:
|
|
|
|
ret = modem_cellular_on_idle_state_leave(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_RESET_PULSE:
|
|
|
|
ret = modem_cellular_on_reset_pulse_state_leave(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_POWER_ON_PULSE:
|
|
|
|
ret = modem_cellular_on_power_on_pulse_state_leave(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_OPEN_DLCI1:
|
|
|
|
ret = modem_cellular_on_open_dlci1_state_leave(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_OPEN_DLCI2:
|
|
|
|
ret = modem_cellular_on_open_dlci2_state_leave(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_RUN_DIAL_SCRIPT:
|
|
|
|
ret = modem_cellular_on_run_dial_script_state_leave(data);
|
|
|
|
break;
|
|
|
|
|
2023-10-31 18:15:48 +01:00
|
|
|
case MODEM_CELLULAR_STATE_AWAIT_REGISTERED:
|
|
|
|
ret = modem_cellular_on_await_registered_state_leave(data);
|
|
|
|
break;
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
case MODEM_CELLULAR_STATE_CARRIER_ON:
|
|
|
|
ret = modem_cellular_on_carrier_on_state_leave(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_INIT_POWER_OFF:
|
|
|
|
ret = modem_cellular_on_init_power_off_state_leave(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_POWER_OFF_PULSE:
|
|
|
|
ret = modem_cellular_on_power_off_pulse_state_leave(data);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_enter_state(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_state state)
|
|
|
|
{
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
ret = modem_cellular_on_state_leave(data);
|
|
|
|
|
|
|
|
if (ret < 0) {
|
|
|
|
LOG_WRN("failed to leave state, error: %i", ret);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
data->state = state;
|
|
|
|
ret = modem_cellular_on_state_enter(data);
|
|
|
|
|
|
|
|
if (ret < 0) {
|
|
|
|
LOG_WRN("failed to enter state error: %i", ret);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_event_handler(struct modem_cellular_data *data,
|
|
|
|
enum modem_cellular_event evt)
|
|
|
|
{
|
|
|
|
enum modem_cellular_state state;
|
|
|
|
|
|
|
|
state = data->state;
|
|
|
|
|
|
|
|
modem_cellular_log_event(evt);
|
|
|
|
|
|
|
|
switch (data->state) {
|
|
|
|
case MODEM_CELLULAR_STATE_IDLE:
|
|
|
|
modem_cellular_idle_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_RESET_PULSE:
|
|
|
|
modem_cellular_reset_pulse_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_POWER_ON_PULSE:
|
|
|
|
modem_cellular_power_on_pulse_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_AWAIT_POWER_ON:
|
|
|
|
modem_cellular_await_power_on_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_RUN_INIT_SCRIPT:
|
|
|
|
modem_cellular_run_init_script_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_CONNECT_CMUX:
|
|
|
|
modem_cellular_connect_cmux_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_OPEN_DLCI1:
|
|
|
|
modem_cellular_open_dlci1_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_OPEN_DLCI2:
|
|
|
|
modem_cellular_open_dlci2_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_RUN_DIAL_SCRIPT:
|
|
|
|
modem_cellular_run_dial_script_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_AWAIT_REGISTERED:
|
|
|
|
modem_cellular_await_registered_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_CARRIER_ON:
|
|
|
|
modem_cellular_carrier_on_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_INIT_POWER_OFF:
|
|
|
|
modem_cellular_init_power_off_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_POWER_OFF_PULSE:
|
|
|
|
modem_cellular_power_off_pulse_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case MODEM_CELLULAR_STATE_AWAIT_POWER_OFF:
|
|
|
|
modem_cellular_await_power_off_event_handler(data, evt);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (state != data->state) {
|
|
|
|
modem_cellular_log_state_changed(state, data->state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void modem_cellular_cmux_handler(struct modem_cmux *cmux, enum modem_cmux_event event,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
struct modem_cellular_data *data = (struct modem_cellular_data *)user_data;
|
|
|
|
|
|
|
|
switch (event) {
|
|
|
|
case MODEM_CMUX_EVENT_CONNECTED:
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_CMUX_CONNECTED);
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef CONFIG_PM_DEVICE
|
|
|
|
static int modem_cellular_pm_action(const struct device *dev, enum pm_device_action action)
|
|
|
|
{
|
|
|
|
struct modem_cellular_data *data = (struct modem_cellular_data *)dev->data;
|
|
|
|
int ret;
|
|
|
|
|
|
|
|
switch (action) {
|
|
|
|
case PM_DEVICE_ACTION_RESUME:
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_RESUME);
|
|
|
|
ret = 0;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case PM_DEVICE_ACTION_SUSPEND:
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_SUSPEND);
|
|
|
|
ret = k_sem_take(&data->suspended_sem, K_SECONDS(30));
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
ret = -ENOTSUP;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
#endif /* CONFIG_PM_DEVICE */
|
|
|
|
|
|
|
|
static int modem_cellular_init(const struct device *dev)
|
|
|
|
{
|
|
|
|
struct modem_cellular_data *data = (struct modem_cellular_data *)dev->data;
|
|
|
|
struct modem_cellular_config *config = (struct modem_cellular_config *)dev->config;
|
|
|
|
|
|
|
|
data->dev = dev;
|
|
|
|
|
|
|
|
k_work_init_delayable(&data->timeout_work, modem_cellular_timeout_handler);
|
|
|
|
|
|
|
|
k_work_init(&data->event_dispatch_work, modem_cellular_event_dispatch_handler);
|
|
|
|
ring_buf_init(&data->event_rb, sizeof(data->event_buf), data->event_buf);
|
|
|
|
|
|
|
|
k_sem_init(&data->suspended_sem, 0, 1);
|
|
|
|
|
|
|
|
if (modem_cellular_gpio_is_enabled(&config->power_gpio)) {
|
|
|
|
gpio_pin_configure_dt(&config->power_gpio, GPIO_OUTPUT_INACTIVE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (modem_cellular_gpio_is_enabled(&config->reset_gpio)) {
|
|
|
|
gpio_pin_configure_dt(&config->reset_gpio, GPIO_OUTPUT_ACTIVE);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
const struct modem_backend_uart_config uart_backend_config = {
|
|
|
|
.uart = config->uart,
|
|
|
|
.receive_buf = data->uart_backend_receive_buf,
|
|
|
|
.receive_buf_size = ARRAY_SIZE(data->uart_backend_receive_buf),
|
|
|
|
.transmit_buf = data->uart_backend_transmit_buf,
|
|
|
|
.transmit_buf_size = ARRAY_SIZE(data->uart_backend_transmit_buf),
|
|
|
|
};
|
|
|
|
|
|
|
|
data->uart_pipe = modem_backend_uart_init(&data->uart_backend,
|
|
|
|
&uart_backend_config);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
const struct modem_cmux_config cmux_config = {
|
|
|
|
.callback = modem_cellular_cmux_handler,
|
|
|
|
.user_data = data,
|
|
|
|
.receive_buf = data->cmux_receive_buf,
|
|
|
|
.receive_buf_size = ARRAY_SIZE(data->cmux_receive_buf),
|
|
|
|
.transmit_buf = data->cmux_transmit_buf,
|
|
|
|
.transmit_buf_size = ARRAY_SIZE(data->cmux_transmit_buf),
|
|
|
|
};
|
|
|
|
|
|
|
|
modem_cmux_init(&data->cmux, &cmux_config);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
const struct modem_cmux_dlci_config dlci1_config = {
|
|
|
|
.dlci_address = 1,
|
|
|
|
.receive_buf = data->dlci1_receive_buf,
|
|
|
|
.receive_buf_size = ARRAY_SIZE(data->dlci1_receive_buf),
|
|
|
|
};
|
|
|
|
|
|
|
|
data->dlci1_pipe = modem_cmux_dlci_init(&data->cmux, &data->dlci1,
|
|
|
|
&dlci1_config);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
const struct modem_cmux_dlci_config dlci2_config = {
|
|
|
|
.dlci_address = 2,
|
|
|
|
.receive_buf = data->dlci2_receive_buf,
|
|
|
|
.receive_buf_size = ARRAY_SIZE(data->dlci2_receive_buf),
|
|
|
|
};
|
|
|
|
|
|
|
|
data->dlci2_pipe = modem_cmux_dlci_init(&data->cmux, &data->dlci2,
|
|
|
|
&dlci2_config);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
const struct modem_chat_config chat_config = {
|
|
|
|
.user_data = data,
|
|
|
|
.receive_buf = data->chat_receive_buf,
|
|
|
|
.receive_buf_size = ARRAY_SIZE(data->chat_receive_buf),
|
|
|
|
.delimiter = data->chat_delimiter,
|
|
|
|
.delimiter_size = ARRAY_SIZE(data->chat_delimiter),
|
|
|
|
.filter = data->chat_filter,
|
|
|
|
.filter_size = ARRAY_SIZE(data->chat_filter),
|
|
|
|
.argv = data->chat_argv,
|
|
|
|
.argv_size = ARRAY_SIZE(data->chat_argv),
|
|
|
|
.unsol_matches = unsol_matches,
|
|
|
|
.unsol_matches_size = ARRAY_SIZE(unsol_matches),
|
|
|
|
.process_timeout = K_MSEC(2),
|
|
|
|
};
|
|
|
|
|
|
|
|
modem_chat_init(&data->chat, &chat_config);
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifndef CONFIG_PM_DEVICE
|
|
|
|
modem_cellular_delegate_event(data, MODEM_CELLULAR_EVENT_RESUME);
|
|
|
|
#else
|
|
|
|
pm_device_init_suspended(dev);
|
|
|
|
#endif /* CONFIG_PM_DEVICE */
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Every modem uses two custom scripts to initialize the modem and dial out.
|
|
|
|
*
|
|
|
|
* The first script is named <dt driver compatible>_init_chat_script, with its
|
|
|
|
* script commands named <dt driver compatible>_init_chat_script_cmds. This
|
|
|
|
* script is sent to the modem after it has started up, and must configure the
|
|
|
|
* modem to use CMUX.
|
|
|
|
*
|
|
|
|
* The second script is named <dt driver compatible>_dial_chat_script, with its
|
|
|
|
* script commands named <dt driver compatible>_dial_chat_script_cmds. This
|
|
|
|
* script is sent on a DLCI channel in command mode, and must request the modem
|
|
|
|
* dial out and put the DLCI channel into data mode.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#if DT_HAS_COMPAT_STATUS_OKAY(quectel_bg95)
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(quectel_bg95_init_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("ATE0", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=4", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CMEE=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGSN", imei_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGMM", cgmm_match),
|
2023-10-31 18:15:48 +01:00
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT+CMUX=0,0,5,127", 300));
|
2023-04-10 18:24:56 +02:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(quectel_bg95_init_chat_script, quectel_bg95_init_chat_script_cmds,
|
|
|
|
abort_matches, modem_cellular_chat_callback_handler, 10);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(quectel_bg95_dial_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_MULT("AT+CGACT=0,1", allow_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGDCONT=1,\"IP\","
|
|
|
|
"\""CONFIG_MODEM_CELLULAR_APN"\"",
|
|
|
|
ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("ATD*99***1#", 0),);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(quectel_bg95_dial_chat_script, quectel_bg95_dial_chat_script_cmds,
|
|
|
|
dial_abort_matches, modem_cellular_chat_callback_handler, 10);
|
2023-10-31 18:15:48 +01:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(quectel_bg95_periodic_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match));
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(quectel_bg95_periodic_chat_script,
|
|
|
|
quectel_bg95_periodic_chat_script_cmds, abort_matches,
|
|
|
|
modem_cellular_chat_callback_handler, 4);
|
2023-04-10 18:24:56 +02:00
|
|
|
#endif
|
|
|
|
|
2023-11-02 17:23:02 +01:00
|
|
|
#if DT_HAS_COMPAT_STATUS_OKAY(quectel_eg25_g)
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(quectel_eg25_g_init_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("ATE0", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=4", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CMEE=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGSN", imei_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGMM", cgmm_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT+CMUX=0,0,5,127,10,3,30,10,2",
|
|
|
|
100));
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(quectel_eg25_g_init_chat_script, quectel_eg25_g_init_chat_script_cmds,
|
|
|
|
abort_matches, modem_cellular_chat_callback_handler, 10);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(quectel_eg25_g_dial_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_MULT("AT+CGACT=0,1", allow_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGDCONT=1,\"IP\","
|
|
|
|
"\""CONFIG_MODEM_CELLULAR_APN"\"",
|
|
|
|
ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("ATD*99***1#", 0),);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(quectel_eg25_g_dial_chat_script, quectel_eg25_g_dial_chat_script_cmds,
|
|
|
|
dial_abort_matches, modem_cellular_chat_callback_handler, 10);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(quectel_eg25_g_periodic_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match));
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(quectel_eg25_g_periodic_chat_script,
|
|
|
|
quectel_eg25_g_periodic_chat_script_cmds, abort_matches,
|
|
|
|
modem_cellular_chat_callback_handler, 4);
|
|
|
|
#endif
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
#if DT_HAS_COMPAT_STATUS_OKAY(zephyr_gsm_ppp)
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(zephyr_gsm_ppp_init_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("ATE0", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=4", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CMEE=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGSN", imei_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGMM", cgmm_match),
|
2023-10-31 18:15:48 +01:00
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
2023-10-02 18:34:28 +02:00
|
|
|
/* The 300ms delay after sending the AT+CMUX command is required
|
|
|
|
* for some modems to ensure they get enough time to enter CMUX
|
|
|
|
* mode before sending the first CMUX command. If this delay is
|
|
|
|
* too short, modems have been observed to simply deadlock,
|
|
|
|
* refusing to respond to any CMUX command.
|
|
|
|
*/
|
2023-10-11 16:14:58 +02:00
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT+CMUX=0,0,5,127", 300));
|
2023-04-10 18:24:56 +02:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(zephyr_gsm_ppp_init_chat_script, zephyr_gsm_ppp_init_chat_script_cmds,
|
|
|
|
abort_matches, modem_cellular_chat_callback_handler, 10);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(zephyr_gsm_ppp_dial_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_MULT("AT+CGACT=0,1", allow_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGDCONT=1,\"IP\","
|
|
|
|
"\""CONFIG_MODEM_CELLULAR_APN"\"",
|
|
|
|
ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("ATD*99***1#", 0),);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(zephyr_gsm_ppp_dial_chat_script, zephyr_gsm_ppp_dial_chat_script_cmds,
|
|
|
|
dial_abort_matches, modem_cellular_chat_callback_handler, 10);
|
2023-10-31 18:15:48 +01:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(zephyr_gsm_ppp_periodic_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match));
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(zephyr_gsm_ppp_periodic_chat_script,
|
|
|
|
zephyr_gsm_ppp_periodic_chat_script_cmds, abort_matches,
|
|
|
|
modem_cellular_chat_callback_handler, 4);
|
2023-04-10 18:24:56 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if DT_HAS_COMPAT_STATUS_OKAY(simcom_sim7080)
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(simcom_sim7080_init_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("ATE0", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=4", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CMEE=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGSN", imei_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGMM", cgmm_match),
|
2023-10-31 18:15:48 +01:00
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
2023-10-11 16:14:58 +02:00
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT+CMUX=0,0,5,127", 300));
|
2023-04-10 18:24:56 +02:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(simcom_sim7080_init_chat_script, simcom_sim7080_init_chat_script_cmds,
|
|
|
|
abort_matches, modem_cellular_chat_callback_handler, 10);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(simcom_sim7080_dial_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_MULT("AT+CGACT=0,1", allow_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGDCONT=1,\"IP\","
|
|
|
|
"\""CONFIG_MODEM_CELLULAR_APN"\"",
|
|
|
|
ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("ATD*99***1#", 0),);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(simcom_sim7080_dial_chat_script, simcom_sim7080_dial_chat_script_cmds,
|
|
|
|
dial_abort_matches, modem_cellular_chat_callback_handler, 10);
|
2023-10-31 18:15:48 +01:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(simcom_sim7080_periodic_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match));
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(simcom_sim7080_periodic_chat_script,
|
|
|
|
simcom_sim7080_periodic_chat_script_cmds, abort_matches,
|
|
|
|
modem_cellular_chat_callback_handler, 4);
|
2023-04-10 18:24:56 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if DT_HAS_COMPAT_STATUS_OKAY(u_blox_sara_r4)
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(u_blox_sara_r4_init_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("ATE0", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=4", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CMEE=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGSN", imei_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGMM", cgmm_match),
|
2023-10-31 18:15:48 +01:00
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CMUX=0,0,5,127", ok_match));
|
2023-04-10 18:24:56 +02:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(u_blox_sara_r4_init_chat_script, u_blox_sara_r4_init_chat_script_cmds,
|
|
|
|
abort_matches, modem_cellular_chat_callback_handler, 10);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(u_blox_sara_r4_dial_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_MULT("AT+CGACT=0,1", allow_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGDCONT=1,\"IP\","
|
|
|
|
"\""CONFIG_MODEM_CELLULAR_APN"\"",
|
|
|
|
ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("ATD*99***1#", 0),);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(u_blox_sara_r4_dial_chat_script, u_blox_sara_r4_dial_chat_script_cmds,
|
|
|
|
dial_abort_matches, modem_cellular_chat_callback_handler, 10);
|
2023-10-31 18:15:48 +01:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(u_blox_sara_r4_periodic_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match));
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(u_blox_sara_r4_periodic_chat_script,
|
|
|
|
u_blox_sara_r4_periodic_chat_script_cmds, abort_matches,
|
|
|
|
modem_cellular_chat_callback_handler, 4);
|
2023-04-10 18:24:56 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#if DT_HAS_COMPAT_STATUS_OKAY(swir_hl7800)
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(swir_hl7800_init_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("ATE0", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_MULT("AT+CGACT=0", allow_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=4", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CMEE=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGSN", imei_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGMM", cgmm_match),
|
2023-10-31 18:15:48 +01:00
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT+CMUX=0,0,5,127", 0));
|
2023-04-10 18:24:56 +02:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(swir_hl7800_init_chat_script, swir_hl7800_init_chat_script_cmds,
|
|
|
|
abort_matches, modem_cellular_chat_callback_handler, 10);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(swir_hl7800_dial_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGDCONT=1,\"IP\","
|
|
|
|
"\""CONFIG_MODEM_CELLULAR_APN"\"",
|
|
|
|
ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+KCNXCFG=1,\"GPRS\",\""
|
|
|
|
CONFIG_MODEM_CELLULAR_APN
|
|
|
|
"\",,,\"IPV4\"",
|
|
|
|
ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("ATD*99***1#", 0));
|
|
|
|
|
2023-10-31 18:15:48 +01:00
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(swir_hl7800_periodic_chat_script_cmds,
|
2023-11-06 14:47:54 +01:00
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match));
|
2023-10-31 18:15:48 +01:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(swir_hl7800_periodic_chat_script,
|
|
|
|
swir_hl7800_periodic_chat_script_cmds, abort_matches,
|
|
|
|
modem_cellular_chat_callback_handler, 4);
|
2023-11-06 14:47:54 +01:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(swir_hl7800_dial_chat_script, swir_hl7800_dial_chat_script_cmds,
|
|
|
|
dial_abort_matches, modem_cellular_chat_callback_handler, 10);
|
2023-04-10 18:24:56 +02:00
|
|
|
#endif
|
|
|
|
|
2023-10-10 15:32:09 +02:00
|
|
|
#if DT_HAS_COMPAT_STATUS_OKAY(telit_me910g1)
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(telit_me910g1_init_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT", 100),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("ATE0", ok_match),
|
|
|
|
/* The Telit me910g1 often has an error trying
|
|
|
|
* to set the PDP context. The radio must be on to set
|
|
|
|
* the context, and this step must be successful.
|
|
|
|
* It is moved to the init script to allow retries.
|
|
|
|
*/
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGDCONT=1,\"IP\","
|
|
|
|
"\"" CONFIG_MODEM_CELLULAR_APN "\"",
|
|
|
|
ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=4", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CMEE=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGSN", imei_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGMM", cgmm_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CFUN=1", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("AT+CMUX=0,0,5,127,10,3,30,10,2",
|
|
|
|
300));
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(telit_me910g1_init_chat_script, telit_me910g1_init_chat_script_cmds,
|
|
|
|
abort_matches, modem_cellular_chat_callback_handler, 10);
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(telit_me910g1_dial_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT", ok_match),
|
2023-11-06 14:47:54 +01:00
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP_NONE("ATD*99***1#", 0));
|
2023-10-10 15:32:09 +02:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(telit_me910g1_dial_chat_script, telit_me910g1_dial_chat_script_cmds,
|
|
|
|
dial_abort_matches, modem_cellular_chat_callback_handler, 10);
|
2023-11-06 14:47:54 +01:00
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_CMDS_DEFINE(telit_me910g1_periodic_chat_script_cmds,
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CGREG?", ok_match),
|
|
|
|
MODEM_CHAT_SCRIPT_CMD_RESP("AT+CEREG?", ok_match));
|
|
|
|
|
|
|
|
MODEM_CHAT_SCRIPT_DEFINE(telit_me910g1_periodic_chat_script,
|
|
|
|
telit_me910g1_periodic_chat_script_cmds, abort_matches,
|
|
|
|
modem_cellular_chat_callback_handler, 4);
|
2023-10-10 15:32:09 +02:00
|
|
|
#endif
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
#define MODEM_CELLULAR_INST_NAME(name, inst) \
|
|
|
|
_CONCAT(_CONCAT(_CONCAT(name, _), DT_DRV_COMPAT), inst)
|
|
|
|
|
2023-10-10 15:17:04 +02:00
|
|
|
#define MODEM_CELLULAR_DEVICE_QUECTEL_BG95(inst) \
|
|
|
|
MODEM_PPP_DEFINE(MODEM_CELLULAR_INST_NAME(ppp, inst), NULL, 98, 1500, 64); \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_data MODEM_CELLULAR_INST_NAME(data, inst) = { \
|
|
|
|
.chat_delimiter = {'\r'}, \
|
|
|
|
.chat_filter = {'\n'}, \
|
|
|
|
.ppp = &MODEM_CELLULAR_INST_NAME(ppp, inst), \
|
|
|
|
}; \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_config MODEM_CELLULAR_INST_NAME(config, inst) = { \
|
|
|
|
.uart = DEVICE_DT_GET(DT_INST_BUS(inst)), \
|
|
|
|
.power_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_power_gpios, {}), \
|
|
|
|
.reset_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_reset_gpios, {}), \
|
|
|
|
.power_pulse_duration_ms = 1500, \
|
|
|
|
.reset_pulse_duration_ms = 100, \
|
|
|
|
.startup_time_ms = 10000, \
|
|
|
|
.shutdown_time_ms = 5000, \
|
|
|
|
.init_chat_script = &quectel_bg95_init_chat_script, \
|
|
|
|
.dial_chat_script = &quectel_bg95_dial_chat_script, \
|
2023-11-06 14:47:54 +01:00
|
|
|
.periodic_chat_script = &quectel_bg95_periodic_chat_script, \
|
2023-10-10 15:17:04 +02:00
|
|
|
}; \
|
|
|
|
\
|
|
|
|
PM_DEVICE_DT_INST_DEFINE(inst, modem_cellular_pm_action); \
|
|
|
|
\
|
|
|
|
DEVICE_DT_INST_DEFINE(inst, modem_cellular_init, PM_DEVICE_DT_INST_GET(inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(data, inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(config, inst), POST_KERNEL, 99, NULL);
|
|
|
|
|
2023-11-02 17:23:02 +01:00
|
|
|
#define MODEM_CELLULAR_DEVICE_QUECTEL_EG25_G(inst) \
|
|
|
|
MODEM_PPP_DEFINE(MODEM_CELLULAR_INST_NAME(ppp, inst), NULL, 98, 1500, 64); \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_data MODEM_CELLULAR_INST_NAME(data, inst) = { \
|
|
|
|
.chat_delimiter = {'\r'}, \
|
|
|
|
.chat_filter = {'\n'}, \
|
|
|
|
.ppp = &MODEM_CELLULAR_INST_NAME(ppp, inst), \
|
|
|
|
}; \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_config MODEM_CELLULAR_INST_NAME(config, inst) = { \
|
|
|
|
.uart = DEVICE_DT_GET(DT_INST_BUS(inst)), \
|
|
|
|
.power_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_power_gpios, {}), \
|
|
|
|
.reset_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_reset_gpios, {}), \
|
|
|
|
.power_pulse_duration_ms = 1500, \
|
|
|
|
.reset_pulse_duration_ms = 500, \
|
|
|
|
.startup_time_ms = 15000, \
|
|
|
|
.shutdown_time_ms = 5000, \
|
|
|
|
.init_chat_script = &quectel_eg25_g_init_chat_script, \
|
|
|
|
.dial_chat_script = &quectel_eg25_g_dial_chat_script, \
|
|
|
|
.periodic_chat_script = &_CONCAT(DT_DRV_COMPAT, _periodic_chat_script), \
|
|
|
|
}; \
|
|
|
|
\
|
|
|
|
PM_DEVICE_DT_INST_DEFINE(inst, modem_cellular_pm_action); \
|
|
|
|
\
|
|
|
|
DEVICE_DT_INST_DEFINE(inst, modem_cellular_init, PM_DEVICE_DT_INST_GET(inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(data, inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(config, inst), POST_KERNEL, 99, NULL);
|
|
|
|
|
2023-10-10 15:17:04 +02:00
|
|
|
#define MODEM_CELLULAR_DEVICE_GSM_PPP(inst) \
|
|
|
|
MODEM_PPP_DEFINE(MODEM_CELLULAR_INST_NAME(ppp, inst), NULL, 98, 1500, 64); \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_data MODEM_CELLULAR_INST_NAME(data, inst) = { \
|
|
|
|
.chat_delimiter = {'\r'}, \
|
|
|
|
.chat_filter = {'\n'}, \
|
|
|
|
.ppp = &MODEM_CELLULAR_INST_NAME(ppp, inst), \
|
|
|
|
}; \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_config MODEM_CELLULAR_INST_NAME(config, inst) = { \
|
|
|
|
.uart = DEVICE_DT_GET(DT_INST_BUS(inst)), \
|
|
|
|
.power_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_power_gpios, {}), \
|
|
|
|
.reset_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_reset_gpios, {}), \
|
|
|
|
.power_pulse_duration_ms = 1500, \
|
|
|
|
.reset_pulse_duration_ms = 100, \
|
|
|
|
.startup_time_ms = 10000, \
|
|
|
|
.shutdown_time_ms = 5000, \
|
|
|
|
.init_chat_script = &zephyr_gsm_ppp_init_chat_script, \
|
|
|
|
.dial_chat_script = &zephyr_gsm_ppp_dial_chat_script, \
|
2023-11-06 14:47:54 +01:00
|
|
|
.periodic_chat_script = &zephyr_gsm_ppp_periodic_chat_script, \
|
2023-10-10 15:17:04 +02:00
|
|
|
}; \
|
|
|
|
\
|
|
|
|
PM_DEVICE_DT_INST_DEFINE(inst, modem_cellular_pm_action); \
|
|
|
|
\
|
|
|
|
DEVICE_DT_INST_DEFINE(inst, modem_cellular_init, PM_DEVICE_DT_INST_GET(inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(data, inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(config, inst), POST_KERNEL, 99, NULL);
|
|
|
|
|
|
|
|
#define MODEM_CELLULAR_DEVICE_SIMCOM_SIM7080(inst) \
|
|
|
|
MODEM_PPP_DEFINE(MODEM_CELLULAR_INST_NAME(ppp, inst), NULL, 98, 1500, 64); \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_data MODEM_CELLULAR_INST_NAME(data, inst) = { \
|
|
|
|
.chat_delimiter = {'\r'}, \
|
|
|
|
.chat_filter = {'\n'}, \
|
|
|
|
.ppp = &MODEM_CELLULAR_INST_NAME(ppp, inst), \
|
|
|
|
}; \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_config MODEM_CELLULAR_INST_NAME(config, inst) = { \
|
|
|
|
.uart = DEVICE_DT_GET(DT_INST_BUS(inst)), \
|
|
|
|
.power_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_power_gpios, {}), \
|
|
|
|
.reset_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_reset_gpios, {}), \
|
|
|
|
.power_pulse_duration_ms = 1500, \
|
|
|
|
.reset_pulse_duration_ms = 100, \
|
|
|
|
.startup_time_ms = 10000, \
|
|
|
|
.shutdown_time_ms = 5000, \
|
|
|
|
.init_chat_script = &simcom_sim7080_init_chat_script, \
|
|
|
|
.dial_chat_script = &simcom_sim7080_dial_chat_script, \
|
2023-11-06 14:47:54 +01:00
|
|
|
.periodic_chat_script = &simcom_sim7080_periodic_chat_script, \
|
2023-10-10 15:17:04 +02:00
|
|
|
}; \
|
|
|
|
\
|
|
|
|
PM_DEVICE_DT_INST_DEFINE(inst, modem_cellular_pm_action); \
|
|
|
|
\
|
|
|
|
DEVICE_DT_INST_DEFINE(inst, modem_cellular_init, PM_DEVICE_DT_INST_GET(inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(data, inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(config, inst), POST_KERNEL, 99, NULL);
|
|
|
|
|
|
|
|
#define MODEM_CELLULAR_DEVICE_U_BLOX_SARA_R4(inst) \
|
|
|
|
MODEM_PPP_DEFINE(MODEM_CELLULAR_INST_NAME(ppp, inst), NULL, 98, 1500, 64); \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_data MODEM_CELLULAR_INST_NAME(data, inst) = { \
|
|
|
|
.chat_delimiter = {'\r'}, \
|
|
|
|
.chat_filter = {'\n'}, \
|
|
|
|
.ppp = &MODEM_CELLULAR_INST_NAME(ppp, inst), \
|
|
|
|
}; \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_config MODEM_CELLULAR_INST_NAME(config, inst) = { \
|
|
|
|
.uart = DEVICE_DT_GET(DT_INST_BUS(inst)), \
|
|
|
|
.power_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_power_gpios, {}), \
|
|
|
|
.reset_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_reset_gpios, {}), \
|
|
|
|
.power_pulse_duration_ms = 1500, \
|
|
|
|
.reset_pulse_duration_ms = 100, \
|
|
|
|
.startup_time_ms = 10000, \
|
|
|
|
.shutdown_time_ms = 5000, \
|
|
|
|
.init_chat_script = &u_blox_sara_r4_init_chat_script, \
|
|
|
|
.dial_chat_script = &u_blox_sara_r4_dial_chat_script, \
|
2023-11-06 14:47:54 +01:00
|
|
|
.periodic_chat_script = &u_blox_sara_r4_periodic_chat_script, \
|
2023-10-10 15:17:04 +02:00
|
|
|
}; \
|
|
|
|
\
|
|
|
|
PM_DEVICE_DT_INST_DEFINE(inst, modem_cellular_pm_action); \
|
|
|
|
\
|
|
|
|
DEVICE_DT_INST_DEFINE(inst, modem_cellular_init, PM_DEVICE_DT_INST_GET(inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(data, inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(config, inst), POST_KERNEL, 99, NULL);
|
|
|
|
|
|
|
|
#define MODEM_CELLULAR_DEVICE_SWIR_HL7800(inst) \
|
|
|
|
MODEM_PPP_DEFINE(MODEM_CELLULAR_INST_NAME(ppp, inst), NULL, 98, 1500, 64); \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_data MODEM_CELLULAR_INST_NAME(data, inst) = { \
|
|
|
|
.chat_delimiter = {'\r'}, \
|
|
|
|
.chat_filter = {'\n'}, \
|
|
|
|
.ppp = &MODEM_CELLULAR_INST_NAME(ppp, inst), \
|
|
|
|
}; \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_config MODEM_CELLULAR_INST_NAME(config, inst) = { \
|
|
|
|
.uart = DEVICE_DT_GET(DT_INST_BUS(inst)), \
|
|
|
|
.power_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_power_gpios, {}), \
|
|
|
|
.reset_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_reset_gpios, {}), \
|
|
|
|
.power_pulse_duration_ms = 1500, \
|
|
|
|
.reset_pulse_duration_ms = 100, \
|
|
|
|
.startup_time_ms = 10000, \
|
|
|
|
.shutdown_time_ms = 5000, \
|
|
|
|
.init_chat_script = &swir_hl7800_init_chat_script, \
|
|
|
|
.dial_chat_script = &swir_hl7800_dial_chat_script, \
|
2023-11-06 14:47:54 +01:00
|
|
|
.periodic_chat_script = &swir_hl7800_periodic_chat_script, \
|
2023-10-10 15:17:04 +02:00
|
|
|
}; \
|
|
|
|
\
|
|
|
|
PM_DEVICE_DT_INST_DEFINE(inst, modem_cellular_pm_action); \
|
|
|
|
\
|
|
|
|
DEVICE_DT_INST_DEFINE(inst, modem_cellular_init, PM_DEVICE_DT_INST_GET(inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(data, inst), \
|
2023-04-10 18:24:56 +02:00
|
|
|
&MODEM_CELLULAR_INST_NAME(config, inst), POST_KERNEL, 99, NULL);
|
|
|
|
|
2023-10-10 15:32:09 +02:00
|
|
|
#define MODEM_CELLULAR_DEVICE_TELIT_ME910G1(inst) \
|
|
|
|
MODEM_PPP_DEFINE(MODEM_CELLULAR_INST_NAME(ppp, inst), NULL, 98, 1500, 64); \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_data MODEM_CELLULAR_INST_NAME(data, inst) = { \
|
|
|
|
.chat_delimiter = {'\r'}, \
|
|
|
|
.chat_filter = {'\n'}, \
|
|
|
|
.ppp = &MODEM_CELLULAR_INST_NAME(ppp, inst), \
|
|
|
|
}; \
|
|
|
|
\
|
|
|
|
static struct modem_cellular_config MODEM_CELLULAR_INST_NAME(config, inst) = { \
|
|
|
|
.uart = DEVICE_DT_GET(DT_INST_BUS(inst)), \
|
|
|
|
.power_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_power_gpios, {}), \
|
|
|
|
.reset_gpio = GPIO_DT_SPEC_INST_GET_OR(inst, mdm_reset_gpios, {}), \
|
|
|
|
.power_pulse_duration_ms = 5050, \
|
|
|
|
.reset_pulse_duration_ms = 250, \
|
|
|
|
.startup_time_ms = 15000, \
|
|
|
|
.shutdown_time_ms = 5000, \
|
|
|
|
.init_chat_script = &telit_me910g1_init_chat_script, \
|
|
|
|
.dial_chat_script = &telit_me910g1_dial_chat_script, \
|
2023-11-06 14:47:54 +01:00
|
|
|
.periodic_chat_script = &telit_me910g1_periodic_chat_script, \
|
2023-10-10 15:32:09 +02:00
|
|
|
}; \
|
|
|
|
\
|
|
|
|
PM_DEVICE_DT_INST_DEFINE(inst, modem_cellular_pm_action); \
|
|
|
|
\
|
|
|
|
DEVICE_DT_INST_DEFINE(inst, modem_cellular_init, PM_DEVICE_DT_INST_GET(inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(data, inst), \
|
|
|
|
&MODEM_CELLULAR_INST_NAME(config, inst), POST_KERNEL, 99, NULL);
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
#define DT_DRV_COMPAT quectel_bg95
|
2023-10-10 15:17:04 +02:00
|
|
|
DT_INST_FOREACH_STATUS_OKAY(MODEM_CELLULAR_DEVICE_QUECTEL_BG95)
|
2023-04-10 18:24:56 +02:00
|
|
|
#undef DT_DRV_COMPAT
|
|
|
|
|
2023-11-02 17:23:02 +01:00
|
|
|
#define DT_DRV_COMPAT quectel_eg25_g
|
|
|
|
DT_INST_FOREACH_STATUS_OKAY(MODEM_CELLULAR_DEVICE_QUECTEL_EG25_G)
|
|
|
|
#undef DT_DRV_COMPAT
|
|
|
|
|
2023-04-10 18:24:56 +02:00
|
|
|
#define DT_DRV_COMPAT zephyr_gsm_ppp
|
2023-10-10 15:17:04 +02:00
|
|
|
DT_INST_FOREACH_STATUS_OKAY(MODEM_CELLULAR_DEVICE_GSM_PPP)
|
2023-04-10 18:24:56 +02:00
|
|
|
#undef DT_DRV_COMPAT
|
|
|
|
|
|
|
|
#define DT_DRV_COMPAT simcom_sim7080
|
2023-10-10 15:17:04 +02:00
|
|
|
DT_INST_FOREACH_STATUS_OKAY(MODEM_CELLULAR_DEVICE_SIMCOM_SIM7080)
|
2023-04-10 18:24:56 +02:00
|
|
|
#undef DT_DRV_COMPAT
|
|
|
|
|
|
|
|
#define DT_DRV_COMPAT u_blox_sara_r4
|
2023-10-10 15:17:04 +02:00
|
|
|
DT_INST_FOREACH_STATUS_OKAY(MODEM_CELLULAR_DEVICE_U_BLOX_SARA_R4)
|
2023-04-10 18:24:56 +02:00
|
|
|
#undef DT_DRV_COMPAT
|
|
|
|
|
|
|
|
#define DT_DRV_COMPAT swir_hl7800
|
2023-10-10 15:17:04 +02:00
|
|
|
DT_INST_FOREACH_STATUS_OKAY(MODEM_CELLULAR_DEVICE_SWIR_HL7800)
|
2023-04-10 18:24:56 +02:00
|
|
|
#undef DT_DRV_COMPAT
|
2023-10-10 15:32:09 +02:00
|
|
|
|
|
|
|
#define DT_DRV_COMPAT telit_me910g1
|
|
|
|
DT_INST_FOREACH_STATUS_OKAY(MODEM_CELLULAR_DEVICE_TELIT_ME910G1)
|
|
|
|
#undef DT_DRV_COMPAT
|