esp_modem.c 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488
  1. // Copyright 2015-2018 Espressif Systems (Shanghai) PTE LTD
  2. //
  3. // Licensed under the Apache License, Version 2.0 (the "License");
  4. // you may not use this file except in compliance with the License.
  5. // You may obtain a copy of the License at
  6. //
  7. // http://www.apache.org/licenses/LICENSE-2.0
  8. //
  9. // Unless required by applicable law or agreed to in writing, software
  10. // distributed under the License is distributed on an "AS IS" BASIS,
  11. // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. // See the License for the specific language governing permissions and
  13. // limitations under the License.
  14. #include <stdlib.h>
  15. #include <string.h>
  16. #include <sys/param.h>
  17. #include "freertos/FreeRTOS.h"
  18. #include "freertos/task.h"
  19. #include "freertos/semphr.h"
  20. #include "esp_modem.h"
  21. #include "esp_log.h"
  22. #include "sdkconfig.h"
  23. #define ESP_MODEM_LINE_BUFFER_SIZE (CONFIG_EXAMPLE_UART_RX_BUFFER_SIZE / 2)
  24. #define ESP_MODEM_EVENT_QUEUE_SIZE (16)
  25. #define MIN_PATTERN_INTERVAL (9)
  26. #define MIN_POST_IDLE (0)
  27. #define MIN_PRE_IDLE (0)
  28. /**
  29. * @brief Macro defined for error checking
  30. *
  31. */
  32. static const char *MODEM_TAG = "esp-modem";
  33. #define MODEM_CHECK(a, str, goto_tag, ...) \
  34. do \
  35. { \
  36. if (!(a)) \
  37. { \
  38. ESP_LOGE(MODEM_TAG, "%s(%d): " str, __FUNCTION__, __LINE__, ##__VA_ARGS__); \
  39. goto goto_tag; \
  40. } \
  41. } while (0)
  42. ESP_EVENT_DEFINE_BASE(ESP_MODEM_EVENT);
  43. /**
  44. * @brief ESP32 Modem DTE
  45. *
  46. */
  47. typedef struct {
  48. uart_port_t uart_port; /*!< UART port */
  49. uint8_t *buffer; /*!< Internal buffer to store response lines/data from DCE */
  50. QueueHandle_t event_queue; /*!< UART event queue handle */
  51. esp_event_loop_handle_t event_loop_hdl; /*!< Event loop handle */
  52. TaskHandle_t uart_event_task_hdl; /*!< UART event task handle */
  53. SemaphoreHandle_t process_sem; /*!< Semaphore used for indicating processing status */
  54. modem_dte_t parent; /*!< DTE interface that should extend */
  55. esp_modem_on_receive receive_cb; /*!< ptr to data reception */
  56. void *receive_cb_ctx; /*!< ptr to rx fn context data */
  57. } esp_modem_dte_t;
  58. esp_err_t esp_modem_set_rx_cb(modem_dte_t *dte, esp_modem_on_receive receive_cb, void *receive_cb_ctx)
  59. {
  60. esp_modem_dte_t *esp_dte = __containerof(dte, esp_modem_dte_t, parent);
  61. esp_dte->receive_cb_ctx = receive_cb_ctx;
  62. esp_dte->receive_cb = receive_cb;
  63. return ESP_OK;
  64. }
  65. /**
  66. * @brief Handle one line in DTE
  67. *
  68. * @param esp_dte ESP modem DTE object
  69. * @return esp_err_t
  70. * - ESP_OK on success
  71. * - ESP_FAIL on error
  72. */
  73. static esp_err_t esp_dte_handle_line(esp_modem_dte_t *esp_dte)
  74. {
  75. modem_dce_t *dce = esp_dte->parent.dce;
  76. MODEM_CHECK(dce, "DTE has not yet bind with DCE", err);
  77. const char *line = (const char *)(esp_dte->buffer);
  78. /* Skip pure "\r\n" lines */
  79. if (strlen(line) > 2) {
  80. MODEM_CHECK(dce->handle_line, "no handler for line", err_handle);
  81. MODEM_CHECK(dce->handle_line(dce, line) == ESP_OK, "handle line failed", err_handle);
  82. }
  83. return ESP_OK;
  84. err_handle:
  85. /* Send ESP_MODEM_EVENT_UNKNOWN signal to event loop */
  86. esp_event_post_to(esp_dte->event_loop_hdl, ESP_MODEM_EVENT, ESP_MODEM_EVENT_UNKNOWN,
  87. (void *)line, strlen(line) + 1, pdMS_TO_TICKS(100));
  88. err:
  89. return ESP_FAIL;
  90. }
  91. /**
  92. * @brief Handle when a pattern has been detected by UART
  93. *
  94. * @param esp_dte ESP32 Modem DTE object
  95. */
  96. static void esp_handle_uart_pattern(esp_modem_dte_t *esp_dte)
  97. {
  98. int pos = uart_pattern_pop_pos(esp_dte->uart_port);
  99. int read_len = 0;
  100. if (pos != -1) {
  101. if (pos < ESP_MODEM_LINE_BUFFER_SIZE - 1) {
  102. /* read one line(include '\n') */
  103. read_len = pos + 1;
  104. } else {
  105. ESP_LOGW(MODEM_TAG, "ESP Modem Line buffer too small");
  106. read_len = ESP_MODEM_LINE_BUFFER_SIZE - 1;
  107. }
  108. read_len = uart_read_bytes(esp_dte->uart_port, esp_dte->buffer, read_len, pdMS_TO_TICKS(100));
  109. if (read_len) {
  110. /* make sure the line is a standard string */
  111. esp_dte->buffer[read_len] = '\0';
  112. /* Send new line to handle */
  113. esp_dte_handle_line(esp_dte);
  114. } else {
  115. ESP_LOGE(MODEM_TAG, "uart read bytes failed");
  116. }
  117. } else {
  118. ESP_LOGW(MODEM_TAG, "Pattern Queue Size too small");
  119. uart_flush(esp_dte->uart_port);
  120. }
  121. }
  122. /**
  123. * @brief Handle when new data received by UART
  124. *
  125. * @param esp_dte ESP32 Modem DTE object
  126. */
  127. static void esp_handle_uart_data(esp_modem_dte_t *esp_dte)
  128. {
  129. size_t length = 0;
  130. uart_get_buffered_data_len(esp_dte->uart_port, &length);
  131. length = MIN(ESP_MODEM_LINE_BUFFER_SIZE, length);
  132. length = uart_read_bytes(esp_dte->uart_port, esp_dte->buffer, length, portMAX_DELAY);
  133. /* pass the input data to configured callback */
  134. if (length) {
  135. esp_dte->receive_cb(esp_dte->buffer, length, esp_dte->receive_cb_ctx);
  136. }
  137. }
  138. /**
  139. * @brief UART Event Task Entry
  140. *
  141. * @param param task parameter
  142. */
  143. static void uart_event_task_entry(void *param)
  144. {
  145. esp_modem_dte_t *esp_dte = (esp_modem_dte_t *)param;
  146. uart_event_t event;
  147. while (1) {
  148. if (xQueueReceive(esp_dte->event_queue, &event, pdMS_TO_TICKS(100))) {
  149. switch (event.type) {
  150. case UART_DATA:
  151. esp_handle_uart_data(esp_dte);
  152. break;
  153. case UART_FIFO_OVF:
  154. ESP_LOGW(MODEM_TAG, "HW FIFO Overflow");
  155. uart_flush_input(esp_dte->uart_port);
  156. xQueueReset(esp_dte->event_queue);
  157. break;
  158. case UART_BUFFER_FULL:
  159. ESP_LOGW(MODEM_TAG, "Ring Buffer Full");
  160. uart_flush_input(esp_dte->uart_port);
  161. xQueueReset(esp_dte->event_queue);
  162. break;
  163. case UART_BREAK:
  164. ESP_LOGW(MODEM_TAG, "Rx Break");
  165. break;
  166. case UART_PARITY_ERR:
  167. ESP_LOGE(MODEM_TAG, "Parity Error");
  168. break;
  169. case UART_FRAME_ERR:
  170. ESP_LOGE(MODEM_TAG, "Frame Error");
  171. break;
  172. case UART_PATTERN_DET:
  173. esp_handle_uart_pattern(esp_dte);
  174. break;
  175. default:
  176. ESP_LOGW(MODEM_TAG, "unknown uart event type: %d", event.type);
  177. break;
  178. }
  179. }
  180. /* Drive the event loop */
  181. esp_event_loop_run(esp_dte->event_loop_hdl, pdMS_TO_TICKS(50));
  182. }
  183. vTaskDelete(NULL);
  184. }
  185. /**
  186. * @brief Send command to DCE
  187. *
  188. * @param dte Modem DTE object
  189. * @param command command string
  190. * @param timeout timeout value, unit: ms
  191. * @return esp_err_t
  192. * - ESP_OK on success
  193. * - ESP_FAIL on error
  194. */
  195. static esp_err_t esp_modem_dte_send_cmd(modem_dte_t *dte, const char *command, uint32_t timeout)
  196. {
  197. esp_err_t ret = ESP_FAIL;
  198. modem_dce_t *dce = dte->dce;
  199. MODEM_CHECK(dce, "DTE has not yet bind with DCE", err);
  200. MODEM_CHECK(command, "command is NULL", err);
  201. esp_modem_dte_t *esp_dte = __containerof(dte, esp_modem_dte_t, parent);
  202. /* Calculate timeout clock tick */
  203. /* Reset runtime information */
  204. dce->state = MODEM_STATE_PROCESSING;
  205. /* Send command via UART */
  206. uart_write_bytes(esp_dte->uart_port, command, strlen(command));
  207. /* Check timeout */
  208. MODEM_CHECK(xSemaphoreTake(esp_dte->process_sem, pdMS_TO_TICKS(timeout)) == pdTRUE, "process command timeout", err);
  209. ret = ESP_OK;
  210. err:
  211. dce->handle_line = NULL;
  212. return ret;
  213. }
  214. /**
  215. * @brief Send data to DCE
  216. *
  217. * @param dte Modem DTE object
  218. * @param data data buffer
  219. * @param length length of data to send
  220. * @return int actual length of data that has been send out
  221. */
  222. static int esp_modem_dte_send_data(modem_dte_t *dte, const char *data, uint32_t length)
  223. {
  224. MODEM_CHECK(data, "data is NULL", err);
  225. esp_modem_dte_t *esp_dte = __containerof(dte, esp_modem_dte_t, parent);
  226. return uart_write_bytes(esp_dte->uart_port, data, length);
  227. err:
  228. return -1;
  229. }
  230. /**
  231. * @brief Send data and wait for prompt from DCE
  232. *
  233. * @param dte Modem DTE object
  234. * @param data data buffer
  235. * @param length length of data to send
  236. * @param prompt pointer of specific prompt
  237. * @param timeout timeout value (unit: ms)
  238. * @return esp_err_t
  239. * ESP_OK on success
  240. * ESP_FAIL on error
  241. */
  242. static esp_err_t esp_modem_dte_send_wait(modem_dte_t *dte, const char *data, uint32_t length,
  243. const char *prompt, uint32_t timeout)
  244. {
  245. MODEM_CHECK(data, "data is NULL", err_param);
  246. MODEM_CHECK(prompt, "prompt is NULL", err_param);
  247. esp_modem_dte_t *esp_dte = __containerof(dte, esp_modem_dte_t, parent);
  248. // We'd better disable pattern detection here for a moment in case prompt string contains the pattern character
  249. uart_disable_pattern_det_intr(esp_dte->uart_port);
  250. // uart_disable_rx_intr(esp_dte->uart_port);
  251. MODEM_CHECK(uart_write_bytes(esp_dte->uart_port, data, length) >= 0, "uart write bytes failed", err_write);
  252. uint32_t len = strlen(prompt);
  253. uint8_t *buffer = calloc(len + 1, sizeof(uint8_t));
  254. int res = uart_read_bytes(esp_dte->uart_port, buffer, len, pdMS_TO_TICKS(timeout));
  255. MODEM_CHECK(res >= len, "wait prompt [%s] timeout", err, prompt);
  256. MODEM_CHECK(!strncmp(prompt, (const char *)buffer, len), "get wrong prompt: %s", err, buffer);
  257. free(buffer);
  258. uart_enable_pattern_det_baud_intr(esp_dte->uart_port, '\n', 1, MIN_PATTERN_INTERVAL, MIN_POST_IDLE, MIN_PRE_IDLE);
  259. return ESP_OK;
  260. err:
  261. free(buffer);
  262. err_write:
  263. uart_enable_pattern_det_baud_intr(esp_dte->uart_port, '\n', 1, MIN_PATTERN_INTERVAL, MIN_POST_IDLE, MIN_PRE_IDLE);
  264. err_param:
  265. return ESP_FAIL;
  266. }
  267. /**
  268. * @brief Change Modem's working mode
  269. *
  270. * @param dte Modem DTE object
  271. * @param new_mode new working mode
  272. * @return esp_err_t
  273. * - ESP_OK on success
  274. * - ESP_FAIL on error
  275. */
  276. static esp_err_t esp_modem_dte_change_mode(modem_dte_t *dte, modem_mode_t new_mode)
  277. {
  278. modem_dce_t *dce = dte->dce;
  279. MODEM_CHECK(dce, "DTE has not yet bind with DCE", err);
  280. esp_modem_dte_t *esp_dte = __containerof(dte, esp_modem_dte_t, parent);
  281. MODEM_CHECK(dce->mode != new_mode, "already in mode: %d", err, new_mode);
  282. switch (new_mode) {
  283. case MODEM_PPP_MODE:
  284. MODEM_CHECK(dce->set_working_mode(dce, new_mode) == ESP_OK, "set new working mode:%d failed", err, new_mode);
  285. uart_disable_pattern_det_intr(esp_dte->uart_port);
  286. uart_enable_rx_intr(esp_dte->uart_port);
  287. break;
  288. case MODEM_COMMAND_MODE:
  289. uart_disable_rx_intr(esp_dte->uart_port);
  290. uart_flush(esp_dte->uart_port);
  291. uart_enable_pattern_det_baud_intr(esp_dte->uart_port, '\n', 1, MIN_PATTERN_INTERVAL, MIN_POST_IDLE, MIN_PRE_IDLE);
  292. uart_pattern_queue_reset(esp_dte->uart_port, CONFIG_EXAMPLE_UART_PATTERN_QUEUE_SIZE);
  293. MODEM_CHECK(dce->set_working_mode(dce, new_mode) == ESP_OK, "set new working mode:%d failed", err, new_mode);
  294. break;
  295. default:
  296. break;
  297. }
  298. return ESP_OK;
  299. err:
  300. return ESP_FAIL;
  301. }
  302. static esp_err_t esp_modem_dte_process_cmd_done(modem_dte_t *dte)
  303. {
  304. esp_modem_dte_t *esp_dte = __containerof(dte, esp_modem_dte_t, parent);
  305. return xSemaphoreGive(esp_dte->process_sem) == pdTRUE ? ESP_OK : ESP_FAIL;
  306. }
  307. /**
  308. * @brief Deinitialize a Modem DTE object
  309. *
  310. * @param dte Modem DTE object
  311. * @return esp_err_t
  312. * - ESP_OK on success
  313. * - ESP_FAIL on error
  314. */
  315. static esp_err_t esp_modem_dte_deinit(modem_dte_t *dte)
  316. {
  317. esp_modem_dte_t *esp_dte = __containerof(dte, esp_modem_dte_t, parent);
  318. /* Delete UART event task */
  319. vTaskDelete(esp_dte->uart_event_task_hdl);
  320. /* Delete semaphore */
  321. vSemaphoreDelete(esp_dte->process_sem);
  322. /* Delete event loop */
  323. esp_event_loop_delete(esp_dte->event_loop_hdl);
  324. /* Uninstall UART Driver */
  325. uart_driver_delete(esp_dte->uart_port);
  326. /* Free memory */
  327. free(esp_dte->buffer);
  328. if (dte->dce) {
  329. dte->dce->dte = NULL;
  330. }
  331. free(esp_dte);
  332. return ESP_OK;
  333. }
  334. modem_dte_t *esp_modem_dte_init(const esp_modem_dte_config_t *config)
  335. {
  336. esp_err_t res;
  337. /* malloc memory for esp_dte object */
  338. esp_modem_dte_t *esp_dte = calloc(1, sizeof(esp_modem_dte_t));
  339. MODEM_CHECK(esp_dte, "calloc esp_dte failed", err_dte_mem);
  340. /* malloc memory to storing lines from modem dce */
  341. esp_dte->buffer = calloc(1, ESP_MODEM_LINE_BUFFER_SIZE);
  342. MODEM_CHECK(esp_dte->buffer, "calloc line memory failed", err_line_mem);
  343. /* Set attributes */
  344. esp_dte->uart_port = config->port_num;
  345. esp_dte->parent.flow_ctrl = config->flow_control;
  346. /* Bind methods */
  347. esp_dte->parent.send_cmd = esp_modem_dte_send_cmd;
  348. esp_dte->parent.send_data = esp_modem_dte_send_data;
  349. esp_dte->parent.send_wait = esp_modem_dte_send_wait;
  350. esp_dte->parent.change_mode = esp_modem_dte_change_mode;
  351. esp_dte->parent.process_cmd_done = esp_modem_dte_process_cmd_done;
  352. esp_dte->parent.deinit = esp_modem_dte_deinit;
  353. /* Config UART */
  354. uart_config_t uart_config = {
  355. .baud_rate = config->baud_rate,
  356. .data_bits = config->data_bits,
  357. .parity = config->parity,
  358. .stop_bits = config->stop_bits,
  359. .source_clk = UART_SCLK_APB,
  360. .flow_ctrl = (config->flow_control == MODEM_FLOW_CONTROL_HW) ? UART_HW_FLOWCTRL_CTS_RTS : UART_HW_FLOWCTRL_DISABLE
  361. };
  362. /* Install UART driver and get event queue used inside driver */
  363. res = uart_driver_install(esp_dte->uart_port, CONFIG_EXAMPLE_UART_RX_BUFFER_SIZE, CONFIG_EXAMPLE_UART_TX_BUFFER_SIZE,
  364. CONFIG_EXAMPLE_UART_EVENT_QUEUE_SIZE, &(esp_dte->event_queue), 0);
  365. MODEM_CHECK(res == ESP_OK, "install uart driver failed", err_uart_config);
  366. MODEM_CHECK(uart_param_config(esp_dte->uart_port, &uart_config) == ESP_OK, "config uart parameter failed", err_uart_config);
  367. if (config->flow_control == MODEM_FLOW_CONTROL_HW) {
  368. res = uart_set_pin(esp_dte->uart_port, CONFIG_EXAMPLE_UART_MODEM_TX_PIN, CONFIG_EXAMPLE_UART_MODEM_RX_PIN,
  369. CONFIG_EXAMPLE_UART_MODEM_RTS_PIN, CONFIG_EXAMPLE_UART_MODEM_CTS_PIN);
  370. } else {
  371. res = uart_set_pin(esp_dte->uart_port, CONFIG_EXAMPLE_UART_MODEM_TX_PIN, CONFIG_EXAMPLE_UART_MODEM_RX_PIN,
  372. UART_PIN_NO_CHANGE, UART_PIN_NO_CHANGE);
  373. }
  374. MODEM_CHECK(res == ESP_OK, "config uart gpio failed", err_uart_config);
  375. /* Set flow control threshold */
  376. if (config->flow_control == MODEM_FLOW_CONTROL_HW) {
  377. res = uart_set_hw_flow_ctrl(esp_dte->uart_port, UART_HW_FLOWCTRL_CTS_RTS, UART_FIFO_LEN - 8);
  378. } else if (config->flow_control == MODEM_FLOW_CONTROL_SW) {
  379. res = uart_set_sw_flow_ctrl(esp_dte->uart_port, true, 8, UART_FIFO_LEN - 8);
  380. }
  381. MODEM_CHECK(res == ESP_OK, "config uart flow control failed", err_uart_config);
  382. /* Set pattern interrupt, used to detect the end of a line. */
  383. res = uart_enable_pattern_det_baud_intr(esp_dte->uart_port, '\n', 1, MIN_PATTERN_INTERVAL, MIN_POST_IDLE, MIN_PRE_IDLE);
  384. /* Set pattern queue size */
  385. res |= uart_pattern_queue_reset(esp_dte->uart_port, CONFIG_EXAMPLE_UART_PATTERN_QUEUE_SIZE);
  386. MODEM_CHECK(res == ESP_OK, "config uart pattern failed", err_uart_pattern);
  387. /* Create Event loop */
  388. esp_event_loop_args_t loop_args = {
  389. .queue_size = ESP_MODEM_EVENT_QUEUE_SIZE,
  390. .task_name = NULL
  391. };
  392. MODEM_CHECK(esp_event_loop_create(&loop_args, &esp_dte->event_loop_hdl) == ESP_OK, "create event loop failed", err_eloop);
  393. /* Create semaphore */
  394. esp_dte->process_sem = xSemaphoreCreateBinary();
  395. MODEM_CHECK(esp_dte->process_sem, "create process semaphore failed", err_sem);
  396. /* Create UART Event task */
  397. BaseType_t ret = xTaskCreate(uart_event_task_entry, //Task Entry
  398. "uart_event", //Task Name
  399. CONFIG_EXAMPLE_UART_EVENT_TASK_STACK_SIZE, //Task Stack Size(Bytes)
  400. esp_dte, //Task Parameter
  401. CONFIG_EXAMPLE_UART_EVENT_TASK_PRIORITY, //Task Priority
  402. & (esp_dte->uart_event_task_hdl) //Task Handler
  403. );
  404. MODEM_CHECK(ret == pdTRUE, "create uart event task failed", err_tsk_create);
  405. return &(esp_dte->parent);
  406. /* Error handling */
  407. err_tsk_create:
  408. vSemaphoreDelete(esp_dte->process_sem);
  409. err_sem:
  410. esp_event_loop_delete(esp_dte->event_loop_hdl);
  411. err_eloop:
  412. uart_disable_pattern_det_intr(esp_dte->uart_port);
  413. err_uart_pattern:
  414. uart_driver_delete(esp_dte->uart_port);
  415. err_uart_config:
  416. free(esp_dte->buffer);
  417. err_line_mem:
  418. free(esp_dte);
  419. err_dte_mem:
  420. return NULL;
  421. }
  422. esp_err_t esp_modem_set_event_handler(modem_dte_t *dte, esp_event_handler_t handler, int32_t event_id, void *handler_args)
  423. {
  424. esp_modem_dte_t *esp_dte = __containerof(dte, esp_modem_dte_t, parent);
  425. return esp_event_handler_register_with(esp_dte->event_loop_hdl, ESP_MODEM_EVENT, event_id, handler, handler_args);
  426. }
  427. esp_err_t esp_modem_remove_event_handler(modem_dte_t *dte, esp_event_handler_t handler)
  428. {
  429. esp_modem_dte_t *esp_dte = __containerof(dte, esp_modem_dte_t, parent);
  430. return esp_event_handler_unregister_with(esp_dte->event_loop_hdl, ESP_MODEM_EVENT, ESP_EVENT_ANY_ID, handler);
  431. }
  432. esp_err_t esp_modem_start_ppp(modem_dte_t *dte)
  433. {
  434. modem_dce_t *dce = dte->dce;
  435. MODEM_CHECK(dce, "DTE has not yet bind with DCE", err);
  436. esp_modem_dte_t *esp_dte = __containerof(dte, esp_modem_dte_t, parent);
  437. /* Set PDP Context */
  438. MODEM_CHECK(dce->define_pdp_context(dce, 1, "IP", CONFIG_EXAMPLE_MODEM_APN) == ESP_OK, "set MODEM APN failed", err);
  439. /* Enter PPP mode */
  440. MODEM_CHECK(dte->change_mode(dte, MODEM_PPP_MODE) == ESP_OK, "enter ppp mode failed", err);
  441. /* post PPP mode started event */
  442. esp_event_post_to(esp_dte->event_loop_hdl, ESP_MODEM_EVENT, ESP_MODEM_EVENT_PPP_START, NULL, 0, 0);
  443. return ESP_OK;
  444. err:
  445. return ESP_FAIL;
  446. }
  447. esp_err_t esp_modem_stop_ppp(modem_dte_t *dte)
  448. {
  449. modem_dce_t *dce = dte->dce;
  450. MODEM_CHECK(dce, "DTE has not yet bind with DCE", err);
  451. /* Enter command mode */
  452. MODEM_CHECK(dte->change_mode(dte, MODEM_COMMAND_MODE) == ESP_OK, "enter command mode failed", err);
  453. /* Hang up */
  454. MODEM_CHECK(dce->hang_up(dce) == ESP_OK, "hang up failed", err);
  455. return ESP_OK;
  456. err:
  457. return ESP_FAIL;
  458. }