ppp.h 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689
  1. /*
  2. * Copyright (c) 2019 Intel Corporation.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_NET_PPP_H_
  7. #define ZEPHYR_INCLUDE_NET_PPP_H_
  8. #include <net/net_if.h>
  9. #include <net/net_pkt.h>
  10. #include <net/net_stats.h>
  11. #include <net/net_mgmt.h>
  12. #ifdef __cplusplus
  13. extern "C" {
  14. #endif
  15. /**
  16. * @brief Point-to-point (PPP) L2/driver support functions
  17. * @defgroup ppp PPP L2/driver Support Functions
  18. * @ingroup networking
  19. * @{
  20. */
  21. /** PPP maximum receive unit (MRU) */
  22. #define PPP_MRU CONFIG_NET_PPP_MTU_MRU
  23. /** PPP maximum transfer unit (MTU) */
  24. #define PPP_MTU PPP_MRU
  25. /** Max length of terminate description string */
  26. #define PPP_MAX_TERMINATE_REASON_LEN 32
  27. /** Length of network interface identifier */
  28. #define PPP_INTERFACE_IDENTIFIER_LEN 8
  29. /** PPP L2 API */
  30. struct ppp_api {
  31. /**
  32. * The net_if_api must be placed in first position in this
  33. * struct so that we are compatible with network interface API.
  34. */
  35. struct net_if_api iface_api;
  36. /** Start the device */
  37. int (*start)(const struct device *dev);
  38. /** Stop the device */
  39. int (*stop)(const struct device *dev);
  40. /** Send a network packet */
  41. int (*send)(const struct device *dev, struct net_pkt *pkt);
  42. #if defined(CONFIG_NET_STATISTICS_PPP)
  43. /** Collect optional PPP specific statistics. This pointer
  44. * should be set by driver if statistics needs to be collected
  45. * for that driver.
  46. */
  47. struct net_stats_ppp *(*get_stats)(const struct device *dev);
  48. #endif
  49. };
  50. /* Make sure that the network interface API is properly setup inside
  51. * PPP API struct (it is the first one).
  52. */
  53. BUILD_ASSERT(offsetof(struct ppp_api, iface_api) == 0);
  54. /**
  55. * PPP protocol types.
  56. * See https://www.iana.org/assignments/ppp-numbers/ppp-numbers.xhtml
  57. * for details.
  58. */
  59. enum ppp_protocol_type {
  60. PPP_IP = 0x0021, /**< RFC 1332 */
  61. PPP_IPV6 = 0x0057, /**< RFC 5072 */
  62. PPP_IPCP = 0x8021, /**< RFC 1332 */
  63. PPP_ECP = 0x8053, /**< RFC 1968 */
  64. PPP_IPV6CP = 0x8057, /**< RFC 5072 */
  65. PPP_CCP = 0x80FD, /**< RFC 1962 */
  66. PPP_LCP = 0xc021, /**< RFC 1661 */
  67. PPP_PAP = 0xc023, /**< RFC 1334 */
  68. PPP_CHAP = 0xc223, /**< RFC 1334 */
  69. PPP_EAP = 0xc227, /**< RFC 2284 */
  70. };
  71. /**
  72. * PPP phases
  73. */
  74. enum ppp_phase {
  75. /** Physical-layer not ready */
  76. PPP_DEAD,
  77. /** Link is being established */
  78. PPP_ESTABLISH,
  79. /** Link authentication with peer */
  80. PPP_AUTH,
  81. /** Network connection establishment */
  82. PPP_NETWORK,
  83. /** Network running */
  84. PPP_RUNNING,
  85. /** Link termination */
  86. PPP_TERMINATE,
  87. };
  88. /**
  89. * PPP states, RFC 1661 ch. 4.2
  90. */
  91. enum ppp_state {
  92. PPP_INITIAL,
  93. PPP_STARTING,
  94. PPP_CLOSED,
  95. PPP_STOPPED,
  96. PPP_CLOSING,
  97. PPP_STOPPING,
  98. PPP_REQUEST_SENT,
  99. PPP_ACK_RECEIVED,
  100. PPP_ACK_SENT,
  101. PPP_OPENED
  102. };
  103. /**
  104. * PPP protocol operations from RFC 1661
  105. */
  106. enum ppp_packet_type {
  107. PPP_CONFIGURE_REQ = 1,
  108. PPP_CONFIGURE_ACK = 2,
  109. PPP_CONFIGURE_NACK = 3,
  110. PPP_CONFIGURE_REJ = 4,
  111. PPP_TERMINATE_REQ = 5,
  112. PPP_TERMINATE_ACK = 6,
  113. PPP_CODE_REJ = 7,
  114. PPP_PROTOCOL_REJ = 8,
  115. PPP_ECHO_REQ = 9,
  116. PPP_ECHO_REPLY = 10,
  117. PPP_DISCARD_REQ = 11
  118. };
  119. /**
  120. * LCP option types from RFC 1661 ch. 6
  121. */
  122. enum lcp_option_type {
  123. LCP_OPTION_RESERVED = 0,
  124. /** Maximum-Receive-Unit */
  125. LCP_OPTION_MRU = 1,
  126. /** Async-Control-Character-Map */
  127. LCP_OPTION_ASYNC_CTRL_CHAR_MAP = 2,
  128. /** Authentication-Protocol */
  129. LCP_OPTION_AUTH_PROTO = 3,
  130. /** Quality-Protocol */
  131. LCP_OPTION_QUALITY_PROTO = 4,
  132. /** Magic-Number */
  133. LCP_OPTION_MAGIC_NUMBER = 5,
  134. /** Protocol-Field-Compression */
  135. LCP_OPTION_PROTO_COMPRESS = 7,
  136. /** Address-and-Control-Field-Compression */
  137. LCP_OPTION_ADDR_CTRL_COMPRESS = 8
  138. } __packed;
  139. /**
  140. * IPCP option types from RFC 1332
  141. */
  142. enum ipcp_option_type {
  143. IPCP_OPTION_RESERVED = 0,
  144. /** IP Addresses */
  145. IPCP_OPTION_IP_ADDRESSES = 1,
  146. /** IP Compression Protocol */
  147. IPCP_OPTION_IP_COMP_PROTO = 2,
  148. /** IP Address */
  149. IPCP_OPTION_IP_ADDRESS = 3,
  150. /* RFC 1877 */
  151. /** Primary DNS Server Address */
  152. IPCP_OPTION_DNS1 = 129,
  153. /** Primary NBNS Server Address */
  154. IPCP_OPTION_NBNS1 = 130,
  155. /** Secondary DNS Server Address */
  156. IPCP_OPTION_DNS2 = 131,
  157. /** Secondary NBNS Server Address */
  158. IPCP_OPTION_NBNS2 = 132,
  159. } __packed;
  160. /**
  161. * IPV6CP option types from RFC 5072
  162. */
  163. enum ipv6cp_option_type {
  164. IPV6CP_OPTION_RESERVED = 0,
  165. /** Interface identifier */
  166. IPV6CP_OPTION_INTERFACE_IDENTIFIER = 1,
  167. } __packed;
  168. /**
  169. * @typedef net_ppp_lcp_echo_reply_cb_t
  170. * @brief A callback function that can be called if a Echo-Reply needs to
  171. * be received.
  172. * @param user_data User settable data that is passed to the callback
  173. * function.
  174. * @param user_data_len Length of the user data.
  175. */
  176. typedef void (*net_ppp_lcp_echo_reply_cb_t)(void *user_data,
  177. size_t user_data_len);
  178. struct ppp_my_option_data;
  179. struct ppp_my_option_info;
  180. /**
  181. * Generic PPP Finite State Machine
  182. */
  183. struct ppp_fsm {
  184. /** Timeout timer */
  185. struct k_work_delayable timer;
  186. struct {
  187. /** Acknowledge Configuration Information */
  188. int (*config_info_ack)(struct ppp_fsm *fsm,
  189. struct net_pkt *pkt,
  190. uint16_t length);
  191. /** Add Configuration Information */
  192. struct net_pkt *(*config_info_add)(struct ppp_fsm *fsm);
  193. /** Length of Configuration Information */
  194. int (*config_info_len)(struct ppp_fsm *fsm);
  195. /** Negative Acknowledge Configuration Information */
  196. int (*config_info_nack)(struct ppp_fsm *fsm,
  197. struct net_pkt *pkt,
  198. uint16_t length,
  199. bool rejected);
  200. /** Request peer's Configuration Information */
  201. int (*config_info_req)(struct ppp_fsm *fsm,
  202. struct net_pkt *pkt,
  203. uint16_t length,
  204. struct net_pkt *ret_pkt);
  205. /** Reject Configuration Information */
  206. int (*config_info_rej)(struct ppp_fsm *fsm,
  207. struct net_pkt *pkt,
  208. uint16_t length);
  209. /** Reset Configuration Information */
  210. void (*config_info_reset)(struct ppp_fsm *fsm);
  211. /** FSM goes to OPENED state */
  212. void (*up)(struct ppp_fsm *fsm);
  213. /** FSM leaves OPENED state */
  214. void (*down)(struct ppp_fsm *fsm);
  215. /** Starting this protocol */
  216. void (*starting)(struct ppp_fsm *fsm);
  217. /** Quitting this protocol */
  218. void (*finished)(struct ppp_fsm *fsm);
  219. /** We received Protocol-Reject */
  220. void (*proto_reject)(struct ppp_fsm *fsm);
  221. /** Retransmit */
  222. void (*retransmit)(struct ppp_fsm *fsm);
  223. /** Any code that is not understood by PPP is passed to
  224. * this FSM for further processing.
  225. */
  226. enum net_verdict (*proto_extension)(struct ppp_fsm *fsm,
  227. enum ppp_packet_type code,
  228. uint8_t id,
  229. struct net_pkt *pkt);
  230. } cb;
  231. struct {
  232. /** Options information */
  233. const struct ppp_my_option_info *info;
  234. /** Options negotiation data */
  235. struct ppp_my_option_data *data;
  236. /** Number of negotiated options */
  237. size_t count;
  238. } my_options;
  239. /** Option bits */
  240. uint32_t flags;
  241. /** Number of re-transmissions left */;
  242. uint32_t retransmits;
  243. /** Number of NACK loops since last ACK */
  244. uint32_t nack_loops;
  245. /** Number of NACKs received */
  246. uint32_t recv_nack_loops;
  247. /** Reason for closing protocol */
  248. char terminate_reason[PPP_MAX_TERMINATE_REASON_LEN];
  249. /** PPP protocol number for this FSM */
  250. uint16_t protocol;
  251. /** Current state of PPP link */
  252. enum ppp_state state;
  253. /** Protocol/layer name of this FSM (for debugging) */
  254. const char *name;
  255. /** Current id */
  256. uint8_t id;
  257. /** Current request id */
  258. uint8_t req_id;
  259. /** Have received valid Ack, Nack or Reject to a Request */
  260. uint8_t ack_received : 1;
  261. };
  262. #define PPP_MY_OPTION_ACKED BIT(0)
  263. #define PPP_MY_OPTION_REJECTED BIT(1)
  264. struct ppp_my_option_data {
  265. uint32_t flags;
  266. };
  267. struct lcp_options {
  268. /** Magic number */
  269. uint32_t magic;
  270. /** Async char map */
  271. uint32_t async_map;
  272. /** Maximum Receive Unit value */
  273. uint16_t mru;
  274. /** Which authentication protocol was negotiated (0 means none) */
  275. uint16_t auth_proto;
  276. };
  277. #if defined(CONFIG_NET_L2_PPP_OPTION_MRU)
  278. #define LCP_NUM_MY_OPTIONS 1
  279. #endif
  280. struct ipcp_options {
  281. /** IPv4 address */
  282. struct in_addr address;
  283. struct in_addr dns1_address;
  284. struct in_addr dns2_address;
  285. };
  286. #define IPCP_NUM_MY_OPTIONS 3
  287. struct ipv6cp_options {
  288. /** Interface identifier */
  289. uint8_t iid[PPP_INTERFACE_IDENTIFIER_LEN];
  290. };
  291. #define IPV6CP_NUM_MY_OPTIONS 1
  292. enum ppp_flags {
  293. PPP_CARRIER_UP,
  294. };
  295. /** PPP L2 context specific to certain network interface */
  296. struct ppp_context {
  297. /** Flags representing PPP state, which are accessed from multiple
  298. * threads.
  299. */
  300. atomic_t flags;
  301. /** PPP startup worker. */
  302. struct k_work_delayable startup;
  303. /** Carrier ON/OFF handler worker. This is used to create
  304. * network interface UP/DOWN event when PPP L2 driver
  305. * notices carrier ON/OFF situation. We must not create another
  306. * network management event from inside management handler thus
  307. * we use worker thread to trigger the UP/DOWN event.
  308. */
  309. struct k_work carrier_work;
  310. struct {
  311. /** Finite state machine for LCP */
  312. struct ppp_fsm fsm;
  313. /** Options that we want to request */
  314. struct lcp_options my_options;
  315. /** Options that peer want to request */
  316. struct lcp_options peer_options;
  317. /** Magic-Number value */
  318. uint32_t magic;
  319. #if defined(CONFIG_NET_L2_PPP_OPTION_MRU)
  320. struct ppp_my_option_data my_options_data[LCP_NUM_MY_OPTIONS];
  321. #endif
  322. } lcp;
  323. #if defined(CONFIG_NET_IPV4)
  324. struct {
  325. /** Finite state machine for IPCP */
  326. struct ppp_fsm fsm;
  327. /** Options that we want to request */
  328. struct ipcp_options my_options;
  329. /** Options that peer want to request */
  330. struct ipcp_options peer_options;
  331. /** My options runtime data */
  332. struct ppp_my_option_data my_options_data[IPCP_NUM_MY_OPTIONS];
  333. } ipcp;
  334. #endif
  335. #if defined(CONFIG_NET_IPV6)
  336. struct {
  337. /** Finite state machine for IPV6CP */
  338. struct ppp_fsm fsm;
  339. /** Options that we want to request */
  340. struct ipv6cp_options my_options;
  341. /** Options that peer want to request */
  342. struct ipv6cp_options peer_options;
  343. /** My options runtime data */
  344. struct ppp_my_option_data my_options_data[IPV6CP_NUM_MY_OPTIONS];
  345. } ipv6cp;
  346. #endif
  347. #if defined(CONFIG_NET_L2_PPP_PAP)
  348. struct {
  349. /** Finite state machine for PAP */
  350. struct ppp_fsm fsm;
  351. } pap;
  352. #endif
  353. #if defined(CONFIG_NET_SHELL)
  354. struct {
  355. struct {
  356. /** Callback to be called when Echo-Reply is received.
  357. */
  358. net_ppp_lcp_echo_reply_cb_t cb;
  359. /** User specific data for the callback */
  360. void *user_data;
  361. /** User data length */
  362. size_t user_data_len;
  363. } echo_reply;
  364. /** Used when waiting Echo-Reply */
  365. struct k_sem wait_echo_reply;
  366. /** Echo-Req data value */
  367. uint32_t echo_req_data;
  368. /** Echo-Reply data value */
  369. uint32_t echo_reply_data;
  370. } shell;
  371. #endif
  372. /** Network interface related to this PPP connection */
  373. struct net_if *iface;
  374. /** Current phase of PPP link */
  375. enum ppp_phase phase;
  376. /** This tells what features the PPP supports. */
  377. enum net_l2_flags ppp_l2_flags;
  378. /** This tells how many network protocols are open */
  379. int network_protos_open;
  380. /** This tells how many network protocols are up */
  381. int network_protos_up;
  382. /** Is network carrier up */
  383. uint16_t is_net_carrier_up : 1;
  384. /** Is PPP ready to receive packets */
  385. uint16_t is_ready_to_serve : 1;
  386. /** Is PPP L2 enabled or not */
  387. uint16_t is_enabled : 1;
  388. /** PPP startup pending */
  389. uint16_t is_startup_pending : 1;
  390. /** PPP enable pending */
  391. uint16_t is_enable_done : 1;
  392. /** IPCP status (up / down) */
  393. uint16_t is_ipcp_up : 1;
  394. /** IPCP open status (open / closed) */
  395. uint16_t is_ipcp_open : 1;
  396. /** IPV6CP status (up / down) */
  397. uint16_t is_ipv6cp_up : 1;
  398. /** IPV6CP open status (open / closed) */
  399. uint16_t is_ipv6cp_open : 1;
  400. /** PAP status (up / down) */
  401. uint16_t is_pap_up : 1;
  402. /** PAP open status (open / closed) */
  403. uint16_t is_pap_open : 1;
  404. };
  405. /**
  406. * @brief Inform PPP L2 driver that carrier is detected.
  407. * This happens when cable is connected etc.
  408. *
  409. * @param iface Network interface
  410. */
  411. void net_ppp_carrier_on(struct net_if *iface);
  412. /**
  413. * @brief Inform PPP L2 driver that carrier was lost.
  414. * This happens when cable is disconnected etc.
  415. *
  416. * @param iface Network interface
  417. */
  418. void net_ppp_carrier_off(struct net_if *iface);
  419. /**
  420. * @brief Initialize PPP L2 stack for a given interface
  421. *
  422. * @param iface A valid pointer to a network interface
  423. */
  424. void net_ppp_init(struct net_if *iface);
  425. /* Management API for PPP */
  426. /** @cond INTERNAL_HIDDEN */
  427. #define PPP_L2_CTX_TYPE struct ppp_context
  428. #define _NET_PPP_LAYER NET_MGMT_LAYER_L2
  429. #define _NET_PPP_CODE 0x209
  430. #define _NET_PPP_BASE (NET_MGMT_IFACE_BIT | \
  431. NET_MGMT_LAYER(_NET_PPP_LAYER) | \
  432. NET_MGMT_LAYER_CODE(_NET_PPP_CODE))
  433. #define _NET_PPP_EVENT (_NET_PPP_BASE | NET_MGMT_EVENT_BIT)
  434. enum net_event_ppp_cmd {
  435. NET_EVENT_PPP_CMD_CARRIER_ON = 1,
  436. NET_EVENT_PPP_CMD_CARRIER_OFF,
  437. NET_EVENT_PPP_CMD_PHASE_RUNNING,
  438. NET_EVENT_PPP_CMD_PHASE_DEAD,
  439. };
  440. #define NET_EVENT_PPP_CARRIER_ON \
  441. (_NET_PPP_EVENT | NET_EVENT_PPP_CMD_CARRIER_ON)
  442. #define NET_EVENT_PPP_CARRIER_OFF \
  443. (_NET_PPP_EVENT | NET_EVENT_PPP_CMD_CARRIER_OFF)
  444. #define NET_EVENT_PPP_PHASE_RUNNING \
  445. (_NET_PPP_EVENT | NET_EVENT_PPP_CMD_PHASE_RUNNING)
  446. #define NET_EVENT_PPP_PHASE_DEAD \
  447. (_NET_PPP_EVENT | NET_EVENT_PPP_CMD_PHASE_DEAD)
  448. struct net_if;
  449. /** @endcond */
  450. /**
  451. * @brief Raise CARRIER_ON event when PPP is connected.
  452. *
  453. * @param iface PPP network interface.
  454. */
  455. #if defined(CONFIG_NET_L2_PPP_MGMT)
  456. void ppp_mgmt_raise_carrier_on_event(struct net_if *iface);
  457. #else
  458. static inline void ppp_mgmt_raise_carrier_on_event(struct net_if *iface)
  459. {
  460. ARG_UNUSED(iface);
  461. }
  462. #endif
  463. /**
  464. * @brief Raise CARRIER_OFF event when PPP is disconnected.
  465. *
  466. * @param iface PPP network interface.
  467. */
  468. #if defined(CONFIG_NET_L2_PPP_MGMT)
  469. void ppp_mgmt_raise_carrier_off_event(struct net_if *iface);
  470. #else
  471. static inline void ppp_mgmt_raise_carrier_off_event(struct net_if *iface)
  472. {
  473. ARG_UNUSED(iface);
  474. }
  475. #endif
  476. /**
  477. * @brief Raise PHASE_RUNNING event when PPP reaching RUNNING phase
  478. *
  479. * @param iface PPP network interface.
  480. */
  481. #if defined(CONFIG_NET_L2_PPP_MGMT)
  482. void ppp_mgmt_raise_phase_running_event(struct net_if *iface);
  483. #else
  484. static inline void ppp_mgmt_raise_phase_running_event(struct net_if *iface)
  485. {
  486. ARG_UNUSED(iface);
  487. }
  488. #endif
  489. /**
  490. * @brief Raise PHASE_DEAD event when PPP reaching DEAD phase
  491. *
  492. * @param iface PPP network interface.
  493. */
  494. #if defined(CONFIG_NET_L2_PPP_MGMT)
  495. void ppp_mgmt_raise_phase_dead_event(struct net_if *iface);
  496. #else
  497. static inline void ppp_mgmt_raise_phase_dead_event(struct net_if *iface)
  498. {
  499. ARG_UNUSED(iface);
  500. }
  501. #endif
  502. /**
  503. * @brief Send PPP Echo-Request to peer. We expect to receive Echo-Reply back.
  504. *
  505. * @param idx PPP network interface index
  506. * @param timeout Amount of time to wait Echo-Reply. The value is in
  507. * milliseconds.
  508. *
  509. * @return 0 if Echo-Reply was received, < 0 if there is a timeout or network
  510. * index is not a valid PPP network index.
  511. */
  512. #if defined(CONFIG_NET_L2_PPP)
  513. int net_ppp_ping(int idx, int32_t timeout);
  514. #else
  515. static inline int net_ppp_ping(int idx, int32_t timeout)
  516. {
  517. ARG_UNUSED(idx);
  518. ARG_UNUSED(timeout);
  519. return -ENOTSUP;
  520. }
  521. #endif
  522. /**
  523. * @brief Get PPP context information. This is only used by net-shell to
  524. * print information about PPP.
  525. *
  526. * @param idx PPP network interface index
  527. *
  528. * @return PPP context or NULL if idx is invalid.
  529. */
  530. #if defined(CONFIG_NET_L2_PPP) && defined(CONFIG_NET_SHELL)
  531. struct ppp_context *net_ppp_context_get(int idx);
  532. #else
  533. static inline struct ppp_context *net_ppp_context_get(int idx)
  534. {
  535. ARG_UNUSED(idx);
  536. return NULL;
  537. }
  538. #endif
  539. #ifdef __cplusplus
  540. }
  541. #endif
  542. /**
  543. * @}
  544. */
  545. #endif /* ZEPHYR_INCLUDE_NET_PPP_H_ */