You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2294 lines
80 KiB
2294 lines
80 KiB
/********************************************************************* |
|
PicoTCP. Copyright (c) 2012 TASS Belgium NV. Some rights reserved. |
|
See LICENSE and COPYING for usage. |
|
|
|
Authors: Serge Gadeyne, Daniele Lacamera, Maxime Vincent |
|
|
|
*********************************************************************/ |
|
|
|
|
|
#include <stdint.h> |
|
#include <stdlib.h> |
|
#include <string.h> |
|
|
|
#include "pico_device.h" |
|
#include "pico_dev_ppp.h" |
|
#include "pico_stack.h" |
|
#include "pico_ipv4.h" |
|
#include "pico_md5.h" |
|
#include "pico_dns_client.h" |
|
|
|
#define ppp_dbg(...) do {} while(0) |
|
/* #define ppp_dbg dbg */ |
|
|
|
/* We should define this in a global header. */ |
|
#define ARRAY_SIZE(arr) (sizeof(arr) / sizeof((arr)[0])) |
|
|
|
#define PICO_PPP_MRU 1514 /* RFC default MRU */ |
|
#define PICO_PPP_MTU 1500 |
|
#define PPP_MAXPKT 2048 |
|
#define PPP_MAX_APN 134 |
|
#define PPP_MAX_USERNAME 134 |
|
#define PPP_MAX_PASSWORD 134 |
|
#define PPP_HDR_SIZE 3u |
|
#define PPP_PROTO_SLOT_SIZE 2u |
|
#define PPP_FCS_SIZE 2u |
|
#define PPP_PROTO_LCP short_be(0xc021) |
|
#define PPP_PROTO_IP short_be(0x0021) |
|
#define PPP_PROTO_PAP short_be(0xc023) |
|
#define PPP_PROTO_CHAP short_be(0xc223) |
|
#define PPP_PROTO_IPCP short_be(0x8021) |
|
|
|
#define PICO_CONF_REQ 1 |
|
#define PICO_CONF_ACK 2 |
|
#define PICO_CONF_NAK 3 |
|
#define PICO_CONF_REJ 4 |
|
#define PICO_CONF_TERM 5 |
|
#define PICO_CONF_TERM_ACK 6 |
|
#define PICO_CONF_CODE_REJ 7 |
|
#define PICO_CONF_PROTO_REJ 8 |
|
#define PICO_CONF_ECHO_REQ 9 |
|
#define PICO_CONF_ECHO_REP 10 |
|
#define PICO_CONF_DISCARD_REQ 11 |
|
|
|
#define LCPOPT_MRU 1u /* param size: 4, fixed: MRU */ |
|
#define LCPOPT_AUTH 3u /* param size: 4-5: AUTH proto */ |
|
#define LCPOPT_QUALITY 4u /* unused for now */ |
|
#define LCPOPT_MAGIC 5u /* param size: 6, fixed: Magic */ |
|
#define LCPOPT_PROTO_COMP 7u /* param size: 0, flag */ |
|
#define LCPOPT_ADDRCTL_COMP 8u /* param size: 0, flag */ |
|
|
|
#define CHAP_MD5_SIZE 16u |
|
#define CHAP_CHALLENGE 1 |
|
#define CHAP_RESPONSE 2 |
|
#define CHAP_SUCCESS 3 |
|
#define CHAP_FAILURE 4 |
|
#define CHALLENGE_SIZE(ppp, ch) ((size_t)((1 + strlen(ppp->password) + short_be((ch)->len)))) |
|
|
|
#define PAP_AUTH_REQ 1 |
|
#define PAP_AUTH_ACK 2 |
|
#define PAP_AUTH_NAK 3 |
|
|
|
|
|
#define PICO_PPP_DEFAULT_TIMER (3) /* seconds */ |
|
#define PICO_PPP_DEFAULT_MAX_TERMINATE (2) |
|
#define PICO_PPP_DEFAULT_MAX_CONFIGURE (10) |
|
#define PICO_PPP_DEFAULT_MAX_FAILURE (5) |
|
#define PICO_PPP_DEFAULT_MAX_DIALTIME (20) |
|
|
|
#define IPCP_ADDR_LEN 6u |
|
#define IPCP_VJ_LEN 6u |
|
#define IPCP_OPT_IP 0x03 |
|
#define IPCP_OPT_VJ 0x02 |
|
#define IPCP_OPT_DNS1 0x81 |
|
#define IPCP_OPT_NBNS1 0x82 |
|
#define IPCP_OPT_DNS2 0x83 |
|
#define IPCP_OPT_NBNS2 0x84 |
|
|
|
static uint8_t LCPOPT_LEN[9] = { |
|
0, 4, 0, 4, 4, 6, 2, 2, 2 |
|
}; |
|
|
|
|
|
/* Protocol defines */ |
|
static const unsigned char AT_S3 = 0x0du; |
|
static const unsigned char AT_S4 = 0x0au; |
|
static const unsigned char PPPF_FLAG_SEQ = 0x7eu; |
|
static const unsigned char PPPF_CTRL_ESC = 0x7du; |
|
static const unsigned char PPPF_ADDR = 0xffu; |
|
static const unsigned char PPPF_CTRL = 0x03u; |
|
|
|
static int ppp_devnum = 0; |
|
static uint8_t ppp_recv_buf[PPP_MAXPKT]; |
|
|
|
PACKED_STRUCT_DEF pico_lcp_hdr { |
|
uint8_t code; |
|
uint8_t id; |
|
uint16_t len; |
|
}; |
|
|
|
PACKED_STRUCT_DEF pico_chap_hdr { |
|
uint8_t code; |
|
uint8_t id; |
|
uint16_t len; |
|
}; |
|
|
|
PACKED_STRUCT_DEF pico_pap_hdr { |
|
uint8_t code; |
|
uint8_t id; |
|
uint16_t len; |
|
}; |
|
|
|
PACKED_STRUCT_DEF pico_ipcp_hdr { |
|
uint8_t code; |
|
uint8_t id; |
|
uint16_t len; |
|
}; |
|
|
|
#ifdef DEBUG_PPP |
|
static int fifo_fd = -1; |
|
#endif |
|
|
|
enum ppp_modem_state { |
|
PPP_MODEM_STATE_INITIAL = 0, |
|
PPP_MODEM_STATE_RESET, |
|
PPP_MODEM_STATE_ECHO, |
|
PPP_MODEM_STATE_CREG, |
|
PPP_MODEM_STATE_CGREG, |
|
PPP_MODEM_STATE_CGDCONT, |
|
PPP_MODEM_STATE_CGATT, |
|
PPP_MODEM_STATE_DIAL, |
|
PPP_MODEM_STATE_CONNECTED, |
|
PPP_MODEM_STATE_MAX |
|
}; |
|
|
|
enum ppp_modem_event { |
|
PPP_MODEM_EVENT_START = 0, |
|
PPP_MODEM_EVENT_STOP, |
|
PPP_MODEM_EVENT_OK, |
|
PPP_MODEM_EVENT_CONNECT, |
|
PPP_MODEM_EVENT_TIMEOUT, |
|
PPP_MODEM_EVENT_MAX |
|
}; |
|
|
|
enum ppp_lcp_state { |
|
PPP_LCP_STATE_INITIAL = 0, |
|
PPP_LCP_STATE_STARTING, |
|
PPP_LCP_STATE_CLOSED, |
|
PPP_LCP_STATE_STOPPED, |
|
PPP_LCP_STATE_CLOSING, |
|
PPP_LCP_STATE_STOPPING, |
|
PPP_LCP_STATE_REQ_SENT, |
|
PPP_LCP_STATE_ACK_RCVD, |
|
PPP_LCP_STATE_ACK_SENT, |
|
PPP_LCP_STATE_OPENED, |
|
PPP_LCP_STATE_MAX |
|
}; |
|
|
|
enum ppp_lcp_event { |
|
PPP_LCP_EVENT_UP = 0, |
|
PPP_LCP_EVENT_DOWN, |
|
PPP_LCP_EVENT_OPEN, |
|
PPP_LCP_EVENT_CLOSE, |
|
PPP_LCP_EVENT_TO_POS, |
|
PPP_LCP_EVENT_TO_NEG, |
|
PPP_LCP_EVENT_RCR_POS, |
|
PPP_LCP_EVENT_RCR_NEG, |
|
PPP_LCP_EVENT_RCA, |
|
PPP_LCP_EVENT_RCN, |
|
PPP_LCP_EVENT_RTR, |
|
PPP_LCP_EVENT_RTA, |
|
PPP_LCP_EVENT_RUC, |
|
PPP_LCP_EVENT_RXJ_POS, |
|
PPP_LCP_EVENT_RXJ_NEG, |
|
PPP_LCP_EVENT_RXR, |
|
PPP_LCP_EVENT_MAX |
|
}; |
|
|
|
enum ppp_auth_state { |
|
PPP_AUTH_STATE_INITIAL = 0, |
|
PPP_AUTH_STATE_STARTING, |
|
PPP_AUTH_STATE_RSP_SENT, |
|
PPP_AUTH_STATE_REQ_SENT, |
|
PPP_AUTH_STATE_AUTHENTICATED, |
|
PPP_AUTH_STATE_MAX |
|
}; |
|
|
|
enum ppp_auth_event { |
|
PPP_AUTH_EVENT_UP_NONE = 0, |
|
PPP_AUTH_EVENT_UP_PAP, |
|
PPP_AUTH_EVENT_UP_CHAP, |
|
PPP_AUTH_EVENT_DOWN, |
|
PPP_AUTH_EVENT_RAC, |
|
PPP_AUTH_EVENT_RAA, |
|
PPP_AUTH_EVENT_RAN, |
|
PPP_AUTH_EVENT_TO, |
|
PPP_AUTH_EVENT_MAX |
|
}; |
|
|
|
enum ppp_ipcp_state { |
|
PPP_IPCP_STATE_INITIAL = 0, |
|
PPP_IPCP_STATE_REQ_SENT, |
|
PPP_IPCP_STATE_ACK_RCVD, |
|
PPP_IPCP_STATE_ACK_SENT, |
|
PPP_IPCP_STATE_OPENED, |
|
PPP_IPCP_STATE_MAX |
|
}; |
|
|
|
enum ppp_ipcp_event { |
|
PPP_IPCP_EVENT_UP = 0, |
|
PPP_IPCP_EVENT_DOWN, |
|
PPP_IPCP_EVENT_RCR_POS, |
|
PPP_IPCP_EVENT_RCR_NEG, |
|
PPP_IPCP_EVENT_RCA, |
|
PPP_IPCP_EVENT_RCN, |
|
PPP_IPCP_EVENT_TO, |
|
PPP_IPCP_EVENT_MAX |
|
}; |
|
|
|
enum pico_ppp_state { |
|
PPP_MODEM_RST = 0, |
|
PPP_MODEM_CREG, |
|
PPP_MODEM_CGREG, |
|
PPP_MODEM_CGDCONT, |
|
PPP_MODEM_CGATT, |
|
PPP_MODEM_CONNECT, |
|
/* From here on, PPP states */ |
|
PPP_ESTABLISH, |
|
PPP_AUTH, |
|
PPP_NETCONFIG, |
|
PPP_NETWORK, |
|
PPP_TERMINATE, |
|
/* MAXSTATE is the last one */ |
|
PPP_MODEM_MAXSTATE |
|
}; |
|
|
|
|
|
#define IPCP_ALLOW_IP 0x01u |
|
#define IPCP_ALLOW_DNS1 0x02u |
|
#define IPCP_ALLOW_DNS2 0x04u |
|
#define IPCP_ALLOW_NBNS1 0x08u |
|
#define IPCP_ALLOW_NBNS2 0x10u |
|
|
|
struct pico_device_ppp { |
|
struct pico_device dev; |
|
int autoreconnect; |
|
enum ppp_modem_state modem_state; |
|
enum ppp_lcp_state lcp_state; |
|
enum ppp_auth_state auth_state; |
|
enum ppp_ipcp_state ipcp_state; |
|
enum pico_ppp_state state; |
|
char apn[PPP_MAX_APN]; |
|
char password[PPP_MAX_PASSWORD]; |
|
char username[PPP_MAX_USERNAME]; |
|
uint16_t lcpopt_local; |
|
uint16_t lcpopt_peer; |
|
uint8_t *pkt; |
|
uint32_t len; |
|
uint16_t rej; |
|
uint16_t auth; |
|
int (*serial_recv)(struct pico_device *dev, void *buf, int len); |
|
int (*serial_send)(struct pico_device *dev, const void *buf, int len); |
|
int (*serial_set_speed)(struct pico_device *dev, uint32_t speed); |
|
uint32_t ipcp_allowed_fields; |
|
uint32_t ipcp_ip; |
|
uint32_t ipcp_dns1; |
|
uint32_t ipcp_nbns1; |
|
uint32_t ipcp_dns2; |
|
uint32_t ipcp_nbns2; |
|
uint32_t timer; |
|
uint8_t timer_val; |
|
uint8_t timer_count; |
|
uint8_t frame_id; |
|
uint8_t timer_on; |
|
uint16_t mru; |
|
}; |
|
|
|
|
|
/* Unit test interceptor */ |
|
static void (*mock_modem_state)(struct pico_device_ppp *ppp, enum ppp_modem_event event) = NULL; |
|
static void (*mock_lcp_state)(struct pico_device_ppp *ppp, enum ppp_lcp_event event) = NULL; |
|
static void (*mock_auth_state)(struct pico_device_ppp *ppp, enum ppp_auth_event event) = NULL; |
|
static void (*mock_ipcp_state)(struct pico_device_ppp *ppp, enum ppp_ipcp_event event) = NULL; |
|
|
|
/* Debug prints */ |
|
#ifdef PPP_DEBUG |
|
static void lcp_optflags_print(struct pico_device_ppp *ppp, uint8_t *opts, uint32_t opts_len); |
|
#endif |
|
|
|
#define PPP_TIMER_ON_MODEM 0x01u |
|
#define PPP_TIMER_ON_LCPREQ 0x04u |
|
#define PPP_TIMER_ON_LCPTERM 0x08u |
|
#define PPP_TIMER_ON_AUTH 0x10u |
|
#define PPP_TIMER_ON_IPCP 0x20u |
|
|
|
/* Escape and send */ |
|
static int ppp_serial_send_escape(struct pico_device_ppp *ppp, void *buf, int len) |
|
{ |
|
uint8_t *in_buf = (uint8_t *)buf; |
|
uint8_t *out_buf = NULL; |
|
int esc_char_count = 0; |
|
int newlen = 0, ret = -1; |
|
int i, j; |
|
|
|
#ifdef PPP_DEBUG |
|
{ |
|
uint32_t idx; |
|
if (len > 0) { |
|
ppp_dbg("PPP >>>> "); |
|
for(idx = 0; idx < (uint32_t)len; idx++) { |
|
ppp_dbg(" %02x", ((uint8_t *)buf)[idx]); |
|
} |
|
ppp_dbg("\n"); |
|
} |
|
} |
|
#endif |
|
|
|
for (i = 1; i < (len - 1); i++) /* from 1 to len -1, as start/stop are not escaped */ |
|
{ |
|
if (((in_buf[i] + 1u) >> 1) == 0x3Fu) |
|
esc_char_count++; |
|
} |
|
if (!esc_char_count) { |
|
return ppp->serial_send(&ppp->dev, buf, len); |
|
} |
|
|
|
newlen = len + esc_char_count; |
|
out_buf = PICO_ZALLOC((uint32_t)newlen); |
|
if(!out_buf) |
|
return -1; |
|
|
|
/* Start byte. */ |
|
out_buf[0] = in_buf[0]; |
|
for(i = 1, j = 1; i < (len - 1); i++) { |
|
if (((in_buf[i] + 1u) >> 1) == 0x3Fu) { |
|
out_buf[j++] = PPPF_CTRL_ESC; |
|
out_buf[j++] = in_buf[i] ^ 0x20; |
|
} else { |
|
out_buf[j++] = in_buf[i]; |
|
} |
|
} |
|
/* Stop byte. */ |
|
out_buf[newlen - 1] = in_buf[len - 1]; |
|
|
|
ret = ppp->serial_send(&ppp->dev, out_buf, newlen); |
|
|
|
PICO_FREE(out_buf); |
|
|
|
if (ret == newlen) |
|
return len; |
|
|
|
return ret; |
|
|
|
} |
|
|
|
static void lcp_timer_start(struct pico_device_ppp *ppp, uint8_t timer_type) |
|
{ |
|
uint8_t count = 0; |
|
ppp->timer_on |= timer_type; |
|
|
|
if (ppp->timer_val == 0) { |
|
ppp->timer_val = PICO_PPP_DEFAULT_TIMER; |
|
} |
|
|
|
if (timer_type == PPP_TIMER_ON_LCPTERM) { |
|
count = PICO_PPP_DEFAULT_MAX_TERMINATE; |
|
} |
|
|
|
if (timer_type == PPP_TIMER_ON_LCPREQ) { |
|
count = PICO_PPP_DEFAULT_MAX_CONFIGURE; |
|
} |
|
|
|
if (timer_type == 0) { |
|
ppp->timer_on |= PPP_TIMER_ON_LCPREQ; |
|
ppp->timer_count = 0; |
|
} |
|
|
|
if (ppp->timer_count == 0) |
|
ppp->timer_count = count; |
|
} |
|
|
|
static void lcp_zero_restart_count(struct pico_device_ppp *ppp) |
|
{ |
|
lcp_timer_start(ppp, 0); |
|
} |
|
|
|
static void lcp_timer_stop(struct pico_device_ppp *ppp, uint8_t timer_type) |
|
{ |
|
ppp->timer_on = (uint8_t)ppp->timer_on & (uint8_t)(~timer_type); |
|
} |
|
|
|
|
|
#define PPP_FSM_MAX_ACTIONS 3 |
|
|
|
struct pico_ppp_fsm { |
|
int next_state; |
|
void (*event_handler[PPP_FSM_MAX_ACTIONS]) (struct pico_device_ppp *); |
|
}; |
|
|
|
#define LCPOPT_SET_LOCAL(ppp, opt) ppp->lcpopt_local |= (uint16_t)(1u << opt) |
|
#define LCPOPT_SET_PEER(ppp, opt) ppp->lcpopt_peer |= (uint16_t)(1u << opt) |
|
#define LCPOPT_UNSET_LOCAL(ppp, opt) ppp->lcpopt_local &= (uint16_t)~(1u << opt) |
|
#define LCPOPT_UNSET_LOCAL_MASK(ppp, opt) ppp->lcpopt_local &= (uint16_t)~(opt) |
|
#define LCPOPT_UNSET_PEER(ppp, opt) ppp->lcpopt_peer &= (uint16_t)~(1u << opt) |
|
#define LCPOPT_ISSET_LOCAL(ppp, opt) ((ppp->lcpopt_local & (uint16_t)(1u << opt)) != 0) |
|
#define LCPOPT_ISSET_PEER(ppp, opt) ((ppp->lcpopt_peer & (uint16_t)(1u << opt)) != 0) |
|
|
|
static void evaluate_modem_state(struct pico_device_ppp *ppp, enum ppp_modem_event event); |
|
static void evaluate_lcp_state(struct pico_device_ppp *ppp, enum ppp_lcp_event event); |
|
static void evaluate_auth_state(struct pico_device_ppp *ppp, enum ppp_auth_event event); |
|
static void evaluate_ipcp_state(struct pico_device_ppp *ppp, enum ppp_ipcp_event event); |
|
|
|
|
|
static uint32_t ppp_ctl_packet_size(struct pico_device_ppp *ppp, uint16_t proto, uint32_t *size) |
|
{ |
|
uint32_t prefix = 0; |
|
|
|
IGNORE_PARAMETER(ppp); |
|
IGNORE_PARAMETER(proto); |
|
|
|
prefix += PPP_HDR_SIZE; /* 7e ff 03 ... */ |
|
prefix += PPP_PROTO_SLOT_SIZE; |
|
*size += prefix; |
|
*size += PPP_FCS_SIZE; |
|
(*size)++; /* STOP byte 0x7e */ |
|
return prefix; |
|
} |
|
|
|
/* CRC16 / FCS Calculation */ |
|
static uint16_t ppp_fcs_char(uint16_t old_crc, uint8_t data) |
|
{ |
|
uint16_t word = (old_crc ^ data) & (uint16_t)0x00FFu; |
|
word = (uint16_t)(word ^ (uint16_t)((word << 4u) & (uint16_t)0x00FFu)); |
|
word = (uint16_t)((word << 8u) ^ (word << 3u) ^ (word >> 4u)); |
|
return ((old_crc >> 8u) ^ word); |
|
} |
|
|
|
static uint16_t ppp_fcs_continue(uint16_t fcs, uint8_t *buf, uint32_t len) |
|
{ |
|
uint8_t *pos = buf; |
|
for (pos = buf; pos < buf + len; pos++) |
|
{ |
|
fcs = ppp_fcs_char(fcs, *pos); |
|
} |
|
return fcs; |
|
} |
|
|
|
static uint16_t ppp_fcs_finish(uint16_t fcs) |
|
{ |
|
return fcs ^ 0xFFFF; |
|
} |
|
|
|
static uint16_t ppp_fcs_start(uint8_t *buf, uint32_t len) |
|
{ |
|
uint16_t fcs = 0xFFFF; |
|
return ppp_fcs_continue(fcs, buf, len); |
|
} |
|
|
|
static int ppp_fcs_verify(uint8_t *buf, uint32_t len) |
|
{ |
|
uint16_t fcs = ppp_fcs_start(buf, len - 2); |
|
fcs = ppp_fcs_finish(fcs); |
|
if ((((fcs & 0xFF00u) >> 8) != buf[len - 1]) || ((fcs & 0xFFu) != buf[len - 2])) { |
|
return -1; |
|
} |
|
|
|
return 0; |
|
} |
|
|
|
/* Serial send (DTE->DCE) functions */ |
|
static int pico_ppp_ctl_send(struct pico_device *dev, uint16_t code, uint8_t *pkt, uint32_t len) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *) dev; |
|
uint16_t fcs; |
|
uint8_t *ptr = pkt; |
|
int i = 0; |
|
|
|
if (!ppp->serial_send) |
|
return (int)len; |
|
|
|
/* PPP Header */ |
|
ptr[i++] = PPPF_FLAG_SEQ; |
|
ptr[i++] = PPPF_ADDR; |
|
ptr[i++] = PPPF_CTRL; |
|
/* protocol */ |
|
ptr[i++] = (uint8_t)(code & 0xFFu); |
|
ptr[i++] = (uint8_t)((code & 0xFF00u) >> 8); |
|
|
|
/* payload is already in place. Calculate FCS. */ |
|
fcs = ppp_fcs_start(pkt + 1, len - 4); /* FCS excludes: start (1), FCS(2), stop(1), total 4 bytes */ |
|
fcs = ppp_fcs_finish(fcs); |
|
pkt[len - 3] = (uint8_t)(fcs & 0xFFu); |
|
pkt[len - 2] = (uint8_t)((fcs & 0xFF00u) >> 8); |
|
pkt[len - 1] = PPPF_FLAG_SEQ; |
|
ppp_serial_send_escape(ppp, pkt, (int)len); |
|
return (int)len; |
|
} |
|
|
|
static uint8_t pico_ppp_data_buffer[PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + PICO_PPP_MTU + PPP_FCS_SIZE + 1]; |
|
static int pico_ppp_send(struct pico_device *dev, void *buf, int len) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *) dev; |
|
uint16_t fcs = 0; |
|
int fcs_start; |
|
int i = 0; |
|
|
|
ppp_dbg(" >>>>>>>>> PPP OUT\n"); |
|
|
|
if (ppp->ipcp_state != PPP_IPCP_STATE_OPENED) |
|
return len; |
|
|
|
if (!ppp->serial_send) |
|
return len; |
|
|
|
pico_ppp_data_buffer[i++] = PPPF_FLAG_SEQ; |
|
if (!LCPOPT_ISSET_PEER(ppp, LCPOPT_ADDRCTL_COMP)) |
|
{ |
|
pico_ppp_data_buffer[i++] = PPPF_ADDR; |
|
pico_ppp_data_buffer[i++] = PPPF_CTRL; |
|
} |
|
|
|
fcs_start = i; |
|
|
|
if (!LCPOPT_ISSET_PEER(ppp, LCPOPT_PROTO_COMP)) |
|
{ |
|
pico_ppp_data_buffer[i++] = 0x00; |
|
} |
|
|
|
pico_ppp_data_buffer[i++] = 0x21; |
|
memcpy(pico_ppp_data_buffer + i, buf, (uint32_t)len); |
|
i += len; |
|
fcs = ppp_fcs_start(pico_ppp_data_buffer + fcs_start, (uint32_t)(i - fcs_start)); |
|
fcs = ppp_fcs_finish(fcs); |
|
pico_ppp_data_buffer[i++] = (uint8_t)(fcs & 0xFFu); |
|
pico_ppp_data_buffer[i++] = (uint8_t)((fcs & 0xFF00u) >> 8); |
|
pico_ppp_data_buffer[i++] = PPPF_FLAG_SEQ; |
|
|
|
ppp_serial_send_escape(ppp, pico_ppp_data_buffer, i); |
|
return len; |
|
} |
|
|
|
|
|
/* FSM functions */ |
|
|
|
static void ppp_modem_start_timer(struct pico_device_ppp *ppp) |
|
{ |
|
ppp->timer_on = ppp->timer_on | PPP_TIMER_ON_MODEM; |
|
ppp->timer_val = PICO_PPP_DEFAULT_TIMER; |
|
} |
|
|
|
#define PPP_AT_CREG0 "ATZ\r\n" |
|
static void ppp_modem_send_reset(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->serial_send) |
|
ppp->serial_send(&ppp->dev, PPP_AT_CREG0, strlen(PPP_AT_CREG0)); |
|
|
|
ppp_modem_start_timer(ppp); |
|
} |
|
|
|
#define PPP_AT_CREG1 "ATE0\r\n" |
|
static void ppp_modem_send_echo(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->serial_send) |
|
ppp->serial_send(&ppp->dev, PPP_AT_CREG1, strlen(PPP_AT_CREG1)); |
|
|
|
ppp_modem_start_timer(ppp); |
|
} |
|
|
|
#define PPP_AT_CREG2 "AT+CREG=1\r\n" |
|
static void ppp_modem_send_creg(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->serial_send) |
|
ppp->serial_send(&ppp->dev, PPP_AT_CREG2, strlen(PPP_AT_CREG2)); |
|
|
|
ppp_modem_start_timer(ppp); |
|
} |
|
|
|
#define PPP_AT_CGREG "AT+CGREG=1\r\n" |
|
static void ppp_modem_send_cgreg(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->serial_send) |
|
ppp->serial_send(&ppp->dev, PPP_AT_CGREG, strlen(PPP_AT_CGREG)); |
|
|
|
ppp_modem_start_timer(ppp); |
|
} |
|
|
|
|
|
#define PPP_AT_CGDCONT "AT+CGDCONT=1,\"IP\",\"%s\",,,\r\n" |
|
static void ppp_modem_send_cgdcont(struct pico_device_ppp *ppp) |
|
{ |
|
char at_cgdcont[200]; |
|
|
|
if (ppp->serial_send) { |
|
snprintf(at_cgdcont, 200, PPP_AT_CGDCONT, ppp->apn); |
|
ppp->serial_send(&ppp->dev, at_cgdcont, (int)strlen(at_cgdcont)); |
|
} |
|
|
|
ppp_modem_start_timer(ppp); |
|
} |
|
|
|
|
|
#define PPP_AT_CGATT "AT+CGATT=1\r\n" |
|
static void ppp_modem_send_cgatt(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->serial_send) |
|
ppp->serial_send(&ppp->dev, PPP_AT_CGATT, strlen(PPP_AT_CGATT)); |
|
|
|
ppp_modem_start_timer(ppp); |
|
} |
|
|
|
#ifdef PICOTCP_PPP_SUPPORT_QUERIES |
|
#define PPP_AT_CGATT_Q "AT+CGATT?\r\n" |
|
static void ppp_modem_send_cgatt_q(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->serial_send) |
|
ppp->serial_send(&ppp->dev, PPP_AT_CGATT_Q, strlen(PPP_AT_CGATT_Q)); |
|
|
|
ppp_modem_start_timer(ppp); |
|
} |
|
#define PPP_AT_CGDCONT_Q "AT+CGDCONT?\r\n" |
|
static void ppp_modem_send_cgdcont_q(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->serial_send) |
|
ppp->serial_send(&ppp->dev, PPP_AT_CGDCONT_Q, strlen(PPP_AT_CGDCONT_Q)); |
|
|
|
ppp_modem_start_timer(ppp); |
|
} |
|
|
|
#define PPP_AT_CGREG_Q "AT+CGREG?\r\n" |
|
static void ppp_modem_send_cgreg_q(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->serial_send) |
|
ppp->serial_send(&ppp->dev, PPP_AT_CGREG_Q, strlen(PPP_AT_CGREG_Q)); |
|
|
|
ppp_modem_start_timer(ppp); |
|
} |
|
|
|
#define PPP_AT_CREG3 "AT+CREG?\r\n" |
|
static void ppp_modem_send_creg_q(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->serial_send) |
|
ppp->serial_send(&ppp->dev, PPP_AT_CREG3, strlen(PPP_AT_CREG3)); |
|
|
|
ppp_modem_start_timer(ppp); |
|
} |
|
#endif /* PICOTCP_PPP_SUPPORT_QUERIES */ |
|
|
|
#define PPP_AT_DIALIN "ATD*99***1#\r\n" |
|
static void ppp_modem_send_dial(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->serial_send) |
|
ppp->serial_send(&ppp->dev, PPP_AT_DIALIN, strlen(PPP_AT_DIALIN)); |
|
|
|
ppp_modem_start_timer(ppp); |
|
ppp->timer_val = PICO_PPP_DEFAULT_MAX_DIALTIME; |
|
} |
|
|
|
static void ppp_modem_connected(struct pico_device_ppp *ppp) |
|
{ |
|
ppp_dbg("PPP: Modem connected to peer.\n"); |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_UP); |
|
} |
|
|
|
#define PPP_ATH "+++ATH\r\n" |
|
static void ppp_modem_disconnected(struct pico_device_ppp *ppp) |
|
{ |
|
ppp_dbg("PPP: Modem disconnected.\n"); |
|
if (ppp->serial_send) |
|
ppp->serial_send(&ppp->dev, PPP_ATH, strlen(PPP_ATH)); |
|
|
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_DOWN); |
|
} |
|
|
|
static const struct pico_ppp_fsm ppp_modem_fsm[PPP_MODEM_STATE_MAX][PPP_MODEM_EVENT_MAX] = { |
|
[PPP_MODEM_STATE_INITIAL] = { |
|
[PPP_MODEM_EVENT_START] = { PPP_MODEM_STATE_RESET, {ppp_modem_send_reset} }, |
|
[PPP_MODEM_EVENT_STOP] = { PPP_MODEM_STATE_INITIAL, {} }, |
|
[PPP_MODEM_EVENT_OK] = { PPP_MODEM_STATE_INITIAL, {} }, |
|
[PPP_MODEM_EVENT_CONNECT] = { PPP_MODEM_STATE_INITIAL, {} }, |
|
[PPP_MODEM_EVENT_TIMEOUT] = { PPP_MODEM_STATE_INITIAL, {ppp_modem_send_reset} } |
|
}, |
|
[PPP_MODEM_STATE_RESET] = { |
|
[PPP_MODEM_EVENT_START] = { PPP_MODEM_STATE_RESET, {} }, |
|
[PPP_MODEM_EVENT_STOP] = { PPP_MODEM_STATE_INITIAL, {} }, |
|
[PPP_MODEM_EVENT_OK] = { PPP_MODEM_STATE_ECHO, { ppp_modem_send_echo } }, |
|
[PPP_MODEM_EVENT_CONNECT] = { PPP_MODEM_STATE_RESET, {} }, |
|
[PPP_MODEM_EVENT_TIMEOUT] = { PPP_MODEM_STATE_RESET, {ppp_modem_send_reset} } |
|
}, |
|
[PPP_MODEM_STATE_ECHO] = { |
|
[PPP_MODEM_EVENT_START] = { PPP_MODEM_STATE_ECHO, {} }, |
|
[PPP_MODEM_EVENT_STOP] = { PPP_MODEM_STATE_INITIAL, {} }, |
|
[PPP_MODEM_EVENT_OK] = { PPP_MODEM_STATE_CREG, { ppp_modem_send_creg } }, |
|
[PPP_MODEM_EVENT_CONNECT] = { PPP_MODEM_STATE_ECHO, {} }, |
|
[PPP_MODEM_EVENT_TIMEOUT] = { PPP_MODEM_STATE_RESET, {ppp_modem_send_reset} } |
|
}, |
|
[PPP_MODEM_STATE_CREG] = { |
|
[PPP_MODEM_EVENT_START] = { PPP_MODEM_STATE_CREG, {} }, |
|
[PPP_MODEM_EVENT_STOP] = { PPP_MODEM_STATE_INITIAL, {} }, |
|
[PPP_MODEM_EVENT_OK] = { PPP_MODEM_STATE_CGREG, { ppp_modem_send_cgreg } }, |
|
[PPP_MODEM_EVENT_CONNECT] = { PPP_MODEM_STATE_CREG, {} }, |
|
[PPP_MODEM_EVENT_TIMEOUT] = { PPP_MODEM_STATE_RESET, {ppp_modem_send_reset} } |
|
}, |
|
[PPP_MODEM_STATE_CGREG] = { |
|
[PPP_MODEM_EVENT_START] = { PPP_MODEM_STATE_CGREG, {} }, |
|
[PPP_MODEM_EVENT_STOP] = { PPP_MODEM_STATE_INITIAL, {} }, |
|
[PPP_MODEM_EVENT_OK] = { PPP_MODEM_STATE_CGDCONT, { ppp_modem_send_cgdcont } }, |
|
[PPP_MODEM_EVENT_CONNECT] = { PPP_MODEM_STATE_CGREG, {} }, |
|
[PPP_MODEM_EVENT_TIMEOUT] = { PPP_MODEM_STATE_RESET, {ppp_modem_send_reset} } |
|
}, |
|
[PPP_MODEM_STATE_CGDCONT] = { |
|
[PPP_MODEM_EVENT_START] = { PPP_MODEM_STATE_CGDCONT, {} }, |
|
[PPP_MODEM_EVENT_STOP] = { PPP_MODEM_STATE_INITIAL, {} }, |
|
[PPP_MODEM_EVENT_OK] = { PPP_MODEM_STATE_CGATT, { ppp_modem_send_cgatt } }, |
|
[PPP_MODEM_EVENT_CONNECT] = { PPP_MODEM_STATE_CGDCONT, {} }, |
|
[PPP_MODEM_EVENT_TIMEOUT] = { PPP_MODEM_STATE_RESET, {ppp_modem_send_reset} } |
|
}, |
|
[PPP_MODEM_STATE_CGATT] = { |
|
[PPP_MODEM_EVENT_START] = { PPP_MODEM_STATE_CGATT, {} }, |
|
[PPP_MODEM_EVENT_STOP] = { PPP_MODEM_STATE_INITIAL, {} }, |
|
[PPP_MODEM_EVENT_OK] = { PPP_MODEM_STATE_DIAL, { ppp_modem_send_dial } }, |
|
[PPP_MODEM_EVENT_CONNECT] = { PPP_MODEM_STATE_CGATT, {} }, |
|
[PPP_MODEM_EVENT_TIMEOUT] = { PPP_MODEM_STATE_RESET, {ppp_modem_send_reset} } |
|
}, |
|
[PPP_MODEM_STATE_DIAL] = { |
|
[PPP_MODEM_EVENT_START] = { PPP_MODEM_STATE_DIAL, {} }, |
|
[PPP_MODEM_EVENT_STOP] = { PPP_MODEM_STATE_INITIAL, {} }, |
|
[PPP_MODEM_EVENT_OK] = { PPP_MODEM_STATE_DIAL, {} }, |
|
[PPP_MODEM_EVENT_CONNECT] = { PPP_MODEM_STATE_CONNECTED, { ppp_modem_connected } }, |
|
[PPP_MODEM_EVENT_TIMEOUT] = { PPP_MODEM_STATE_RESET, {ppp_modem_send_reset} } |
|
}, |
|
[PPP_MODEM_STATE_CONNECTED] = { |
|
[PPP_MODEM_EVENT_START] = { PPP_MODEM_STATE_CONNECTED, {} }, |
|
[PPP_MODEM_EVENT_STOP] = { PPP_MODEM_STATE_INITIAL, { ppp_modem_disconnected } }, |
|
[PPP_MODEM_EVENT_OK] = { PPP_MODEM_STATE_CONNECTED, {} }, |
|
[PPP_MODEM_EVENT_CONNECT] = { PPP_MODEM_STATE_CONNECTED, {} }, |
|
[PPP_MODEM_EVENT_TIMEOUT] = { PPP_MODEM_STATE_CONNECTED, {} } |
|
} |
|
}; |
|
static void evaluate_modem_state(struct pico_device_ppp *ppp, enum ppp_modem_event event) |
|
{ |
|
const struct pico_ppp_fsm *fsm; |
|
int i; |
|
if (mock_modem_state) { |
|
mock_modem_state(ppp, event); |
|
return; |
|
} |
|
|
|
fsm = &ppp_modem_fsm[ppp->modem_state][event]; |
|
|
|
ppp->modem_state = (enum ppp_modem_state)fsm->next_state; |
|
|
|
for (i = 0; i < PPP_FSM_MAX_ACTIONS; i++) { |
|
if (fsm->event_handler[i]) |
|
fsm->event_handler[i](ppp); |
|
} |
|
} |
|
|
|
static void ppp_modem_recv(struct pico_device_ppp *ppp, void *data, uint32_t len) |
|
{ |
|
IGNORE_PARAMETER(len); |
|
|
|
ppp_dbg("PPP: Recv: '%s'\n", (char *)data); |
|
|
|
if (strcmp(data, "OK") == 0) { |
|
evaluate_modem_state(ppp, PPP_MODEM_EVENT_OK); |
|
} |
|
|
|
if (strcmp(data, "ERROR") == 0) { |
|
evaluate_modem_state(ppp, PPP_MODEM_EVENT_STOP); |
|
} |
|
|
|
if (strncmp(data, "CONNECT", 7) == 0) { |
|
evaluate_modem_state(ppp, PPP_MODEM_EVENT_CONNECT); |
|
} |
|
} |
|
|
|
static void lcp_send_configure_request(struct pico_device_ppp *ppp) |
|
{ |
|
# define MY_LCP_REQ_SIZE 12 /* Max value. */ |
|
struct pico_lcp_hdr *req; |
|
uint8_t *lcpbuf, *opts; |
|
uint32_t size = MY_LCP_REQ_SIZE; |
|
uint32_t prefix; |
|
uint32_t optsize = 0; |
|
|
|
prefix = ppp_ctl_packet_size(ppp, PPP_PROTO_LCP, &size); |
|
lcpbuf = PICO_ZALLOC(size); |
|
if (!lcpbuf) |
|
return; |
|
|
|
req = (struct pico_lcp_hdr *)(lcpbuf + prefix); |
|
|
|
opts = lcpbuf + prefix + (sizeof(struct pico_lcp_hdr)); |
|
/* uint8_t my_pkt[] = { 0x7e, 0xff, 0x03, 0xc0, 0x21, 0x01, 0x00, 0x00, 0x06, 0x07, 0x02, 0x64, 0x7b, 0x7e }; */ |
|
|
|
ppp_dbg("Sending LCP CONF REQ\n"); |
|
req->code = PICO_CONF_REQ; |
|
req->id = ppp->frame_id++; |
|
|
|
if (LCPOPT_ISSET_LOCAL(ppp, LCPOPT_PROTO_COMP)) { |
|
opts[optsize++] = LCPOPT_PROTO_COMP; |
|
opts[optsize++] = LCPOPT_LEN[LCPOPT_PROTO_COMP]; |
|
} |
|
|
|
if (LCPOPT_ISSET_LOCAL(ppp, LCPOPT_MRU)) { |
|
opts[optsize++] = LCPOPT_MRU; |
|
opts[optsize++] = LCPOPT_LEN[LCPOPT_MRU]; |
|
opts[optsize++] = (uint8_t)((ppp->mru >> 8) & 0xFF); |
|
opts[optsize++] = (uint8_t)(ppp->mru & 0xFF); |
|
} else { |
|
ppp->mru = PICO_PPP_MRU; |
|
} |
|
|
|
if (LCPOPT_ISSET_LOCAL(ppp, LCPOPT_ADDRCTL_COMP)) { |
|
opts[optsize++] = LCPOPT_ADDRCTL_COMP; |
|
opts[optsize++] = LCPOPT_LEN[LCPOPT_ADDRCTL_COMP]; |
|
} |
|
|
|
req->len = short_be((uint16_t)((unsigned long)optsize + sizeof(struct pico_lcp_hdr))); |
|
|
|
#ifdef PPP_DEBUG |
|
lcp_optflags_print(ppp, opts, optsize); |
|
#endif |
|
|
|
pico_ppp_ctl_send(&ppp->dev, PPP_PROTO_LCP, |
|
lcpbuf, /* Start of PPP packet */ |
|
(uint32_t)(prefix + /* PPP Header, etc. */ |
|
sizeof(struct pico_lcp_hdr) + /* LCP HDR */ |
|
optsize + /* Actual options size */ |
|
PPP_FCS_SIZE + /* FCS at the end of the frame */ |
|
1u) /* STOP Byte */ |
|
); |
|
PICO_FREE(lcpbuf); |
|
ppp->timer_val = PICO_PPP_DEFAULT_TIMER; |
|
lcp_timer_start(ppp, PPP_TIMER_ON_LCPREQ); |
|
} |
|
|
|
#ifdef PPP_DEBUG |
|
static void lcp_optflags_print(struct pico_device_ppp *ppp, uint8_t *opts, uint32_t opts_len) |
|
{ |
|
uint8_t *p = opts; |
|
int off; |
|
IGNORE_PARAMETER(ppp); |
|
ppp_dbg("Parsing options:\n"); |
|
while(p < (opts + opts_len)) { |
|
int i; |
|
|
|
ppp_dbg("-- LCP opt: %d - len: %d - data:", p[0], p[1]); |
|
for (i=0; i<p[1]-2; i++) |
|
{ |
|
ppp_dbg(" %02X", p[2+i]); |
|
} |
|
ppp_dbg("\n"); |
|
|
|
off = p[1]; |
|
if (!off) |
|
break; |
|
|
|
p += off; |
|
} |
|
} |
|
#endif |
|
|
|
/* setting adjust_opts will adjust our options to the ones supplied */ |
|
static uint16_t lcp_optflags(struct pico_device_ppp *ppp, uint8_t *pkt, uint32_t len, int adjust_opts) |
|
{ |
|
uint16_t flags = 0; |
|
uint8_t *p = pkt + sizeof(struct pico_lcp_hdr); |
|
int off; |
|
while(p < (pkt + len)) { |
|
flags = (uint16_t)((uint16_t)(1u << (uint16_t)p[0]) | flags); |
|
|
|
if (adjust_opts && ppp) |
|
{ |
|
switch (p[0]) |
|
{ |
|
case LCPOPT_MRU: |
|
//XXX: Can we accept any MRU ? |
|
ppp_dbg("Adjusting MRU to %02x%02x\n", p[2], p[3]); |
|
ppp->mru = (uint16_t)((p[2] << 8) + p[3]); |
|
break; |
|
case LCPOPT_AUTH: |
|
ppp_dbg("Setting AUTH to %02x%02x\n", p[2], p[3]); |
|
ppp->auth = (uint16_t)((p[2] << 8) + p[3]); |
|
break; |
|
default: |
|
break; |
|
} |
|
} |
|
|
|
off = p[1]; /* opt length field */ |
|
if (!off) |
|
break; |
|
|
|
p += off; |
|
} |
|
#ifdef PPP_DEBUG |
|
lcp_optflags_print(ppp, pkt + sizeof(struct pico_lcp_hdr), (uint32_t)(len - sizeof(struct pico_lcp_hdr)) ); |
|
#endif |
|
return flags; |
|
} |
|
|
|
|
|
static void lcp_send_configure_ack(struct pico_device_ppp *ppp) |
|
{ |
|
uint8_t ack[ppp->len + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_lcp_hdr) + PPP_FCS_SIZE + 1]; |
|
struct pico_lcp_hdr *ack_hdr = (struct pico_lcp_hdr *) (ack + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE); |
|
struct pico_lcp_hdr *lcpreq = (struct pico_lcp_hdr *)ppp->pkt; |
|
memcpy(ack + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE, ppp->pkt, ppp->len); |
|
ack_hdr->code = PICO_CONF_ACK; |
|
ack_hdr->id = lcpreq->id; |
|
ack_hdr->len = lcpreq->len; |
|
ppp_dbg("Sending LCP CONF ACK\n"); |
|
pico_ppp_ctl_send(&ppp->dev, PPP_PROTO_LCP, ack, |
|
PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + /* PPP Header, etc. */ |
|
short_be(lcpreq->len) + /* Actual options size + hdr (whole lcp packet) */ |
|
PPP_FCS_SIZE + /* FCS at the end of the frame */ |
|
1 /* STOP Byte */ |
|
); |
|
} |
|
|
|
static void lcp_send_terminate_request(struct pico_device_ppp *ppp) |
|
{ |
|
uint8_t term[PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_lcp_hdr) + PPP_FCS_SIZE + 1]; |
|
struct pico_lcp_hdr *term_hdr = (struct pico_lcp_hdr *) (term + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE); |
|
term_hdr->code = PICO_CONF_TERM; |
|
term_hdr->id = ppp->frame_id++; |
|
term_hdr->len = short_be((uint16_t)sizeof(struct pico_lcp_hdr)); |
|
ppp_dbg("Sending LCP TERMINATE REQUEST\n"); |
|
pico_ppp_ctl_send(&ppp->dev, PPP_PROTO_LCP, term, |
|
PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + /* PPP Header, etc. */ |
|
sizeof(struct pico_lcp_hdr) + /* Actual options size + hdr (whole lcp packet) */ |
|
PPP_FCS_SIZE + /* FCS at the end of the frame */ |
|
1 /* STOP Byte */ |
|
); |
|
lcp_timer_start(ppp, PPP_TIMER_ON_LCPTERM); |
|
} |
|
|
|
static void lcp_send_terminate_ack(struct pico_device_ppp *ppp) |
|
{ |
|
uint8_t ack[ppp->len + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_lcp_hdr) + PPP_FCS_SIZE + 1]; |
|
struct pico_lcp_hdr *ack_hdr = (struct pico_lcp_hdr *) (ack + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE); |
|
struct pico_lcp_hdr *lcpreq = (struct pico_lcp_hdr *)ppp->pkt; |
|
memcpy(ack + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE, ppp->pkt, ppp->len); |
|
ack_hdr->code = PICO_CONF_TERM_ACK; |
|
ack_hdr->id = lcpreq->id; |
|
ack_hdr->len = lcpreq->len; |
|
ppp_dbg("Sending LCP TERM ACK\n"); |
|
pico_ppp_ctl_send(&ppp->dev, PPP_PROTO_LCP, ack, |
|
PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + /* PPP Header, etc. */ |
|
short_be(lcpreq->len) + /* Actual options size + hdr (whole lcp packet) */ |
|
PPP_FCS_SIZE + /* FCS at the end of the frame */ |
|
1 /* STOP Byte */ |
|
); |
|
} |
|
|
|
static void lcp_send_configure_nack(struct pico_device_ppp *ppp) |
|
{ |
|
uint8_t reject[64]; |
|
uint8_t *p = ppp->pkt + sizeof(struct pico_lcp_hdr); |
|
struct pico_lcp_hdr *lcpreq = (struct pico_lcp_hdr *)ppp->pkt; |
|
struct pico_lcp_hdr *lcprej = (struct pico_lcp_hdr *)(reject + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE); |
|
uint8_t *dst_opts = reject + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_lcp_hdr); |
|
uint32_t dstopts_len = 0; |
|
ppp_dbg("CONF_NACK: rej = %04X\n", ppp->rej); |
|
while (p < (ppp->pkt + ppp->len)) { |
|
uint8_t i = 0; |
|
if ((1u << p[0]) & ppp->rej || (p[0] > 8u)) { /* Reject anything we dont support or with option id >8 */ |
|
ppp_dbg("rejecting option %d -- ", p[0]); |
|
dst_opts[dstopts_len++] = p[0]; |
|
|
|
ppp_dbg("len: %d -- ", p[1]); |
|
dst_opts[dstopts_len++] = p[1]; |
|
|
|
ppp_dbg("data: "); |
|
for(i = 0; i < p[1]-2u; i++) { /* length includes type, length and data fields */ |
|
dst_opts[dstopts_len++] = p[2 + i]; |
|
ppp_dbg("%02X ", p[2+i]); |
|
} |
|
ppp_dbg("\n"); |
|
} |
|
|
|
p += p[1]; |
|
} |
|
lcprej->code = PICO_CONF_REJ; |
|
lcprej->id = lcpreq->id; |
|
lcprej->len = short_be((uint16_t)(dstopts_len + sizeof(struct pico_lcp_hdr))); |
|
|
|
ppp_dbg("Sending LCP CONF REJ\n"); |
|
#ifdef PPP_DEBUG |
|
lcp_optflags_print(ppp, dst_opts, dstopts_len); |
|
#endif |
|
|
|
pico_ppp_ctl_send(&ppp->dev, PPP_PROTO_LCP, reject, |
|
(uint32_t)(PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + /* PPP Header, etc. */ |
|
sizeof(struct pico_lcp_hdr) + /* LCP HDR */ |
|
dstopts_len + /* Actual options size */ |
|
PPP_FCS_SIZE + /* FCS at the end of the frame */ |
|
1u) /* STOP Byte */ |
|
); |
|
} |
|
|
|
static void lcp_process_in(struct pico_device_ppp *ppp, uint8_t *pkt, uint32_t len) |
|
{ |
|
uint16_t optflags; |
|
if (!ppp) |
|
return; |
|
if (pkt[0] == PICO_CONF_REQ) { |
|
uint16_t rejected = 0; |
|
ppp_dbg("Received LCP CONF REQ\n"); |
|
optflags = lcp_optflags(ppp, pkt, len, 1u); |
|
rejected = (uint16_t)(optflags & (~ppp->lcpopt_local)); |
|
ppp->pkt = pkt; |
|
ppp->len = len; |
|
ppp->rej = rejected; |
|
if (rejected) { |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_RCR_NEG); |
|
} else { |
|
ppp->lcpopt_peer = optflags; |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_RCR_POS); |
|
} |
|
|
|
return; |
|
} |
|
|
|
if (pkt[0] == PICO_CONF_ACK) { |
|
ppp_dbg("Received LCP CONF ACK\nOptflags: %04x\n", lcp_optflags(NULL, pkt, len, 0u)); |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_RCA); |
|
return; |
|
} |
|
|
|
if (pkt[0] == PICO_CONF_NAK) { |
|
/* Every instance of the received Configuration Options is recognizable, but some values are not acceptable */ |
|
optflags = lcp_optflags(ppp, pkt, len, 1u); /* We want our options adjusted */ |
|
ppp_dbg("Received LCP CONF NAK - changed optflags: %04X\n", optflags); |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_RCN); |
|
return; |
|
} |
|
|
|
if (pkt[0] == PICO_CONF_REJ) { |
|
/* Some Configuration Options received in a Configure-Request are not recognizable or are not acceptable for negotiation */ |
|
optflags = lcp_optflags(ppp, pkt, len, 0u); |
|
ppp_dbg("Received LCP CONF REJ - will disable optflags: %04X\n", optflags); |
|
/* Disable the options that are not supported by the peer */ |
|
LCPOPT_UNSET_LOCAL_MASK(ppp, optflags); |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_RCN); |
|
return; |
|
} |
|
|
|
if (pkt[0] == PICO_CONF_ECHO_REQ) { |
|
ppp_dbg("Received LCP ECHO REQ\n"); |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_RXR); |
|
return; |
|
} |
|
} |
|
|
|
static void pap_process_in(struct pico_device_ppp *ppp, uint8_t *pkt, uint32_t len) |
|
{ |
|
struct pico_pap_hdr *p = (struct pico_pap_hdr *)pkt; |
|
(void)len; |
|
if (!p) |
|
return; |
|
|
|
if (ppp->auth != 0xc023) |
|
return; |
|
|
|
switch(p->code) { |
|
case PAP_AUTH_ACK: |
|
ppp_dbg("PAP: Received Authentication OK!\n"); |
|
evaluate_auth_state(ppp, PPP_AUTH_EVENT_RAA); |
|
break; |
|
case PAP_AUTH_NAK: |
|
ppp_dbg("PAP: Received Authentication Reject!\n"); |
|
evaluate_auth_state(ppp, PPP_AUTH_EVENT_RAN); |
|
break; |
|
|
|
default: |
|
ppp_dbg("PAP: Received invalid packet with code %d\n", p->code); |
|
} |
|
} |
|
|
|
|
|
static void chap_process_in(struct pico_device_ppp *ppp, uint8_t *pkt, uint32_t len) |
|
{ |
|
struct pico_chap_hdr *ch = (struct pico_chap_hdr *)pkt; |
|
|
|
if (!pkt) |
|
return; |
|
|
|
if (ppp->auth != 0xc223) |
|
return; |
|
|
|
switch(ch->code) { |
|
case CHAP_CHALLENGE: |
|
ppp_dbg("Received CHAP CHALLENGE\n"); |
|
ppp->pkt = pkt; |
|
ppp->len = len; |
|
evaluate_auth_state(ppp, PPP_AUTH_EVENT_RAC); |
|
break; |
|
case CHAP_SUCCESS: |
|
ppp_dbg("Received CHAP SUCCESS\n"); |
|
evaluate_auth_state(ppp, PPP_AUTH_EVENT_RAA); |
|
break; |
|
case CHAP_FAILURE: |
|
ppp_dbg("Received CHAP FAILURE\n"); |
|
evaluate_auth_state(ppp, PPP_AUTH_EVENT_RAN); |
|
break; |
|
} |
|
} |
|
|
|
|
|
static void ipcp_send_ack(struct pico_device_ppp *ppp) |
|
{ |
|
uint8_t ack[ppp->len + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_lcp_hdr) + PPP_FCS_SIZE + 1]; |
|
struct pico_ipcp_hdr *ack_hdr = (struct pico_ipcp_hdr *) (ack + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE); |
|
struct pico_ipcp_hdr *ipcpreq = (struct pico_ipcp_hdr *)ppp->pkt; |
|
memcpy(ack + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE, ppp->pkt, ppp->len); |
|
ack_hdr->code = PICO_CONF_ACK; |
|
ack_hdr->id = ipcpreq->id; |
|
ack_hdr->len = ipcpreq->len; |
|
ppp_dbg("Sending IPCP CONF ACK\n"); |
|
pico_ppp_ctl_send(&ppp->dev, PPP_PROTO_IPCP, ack, |
|
PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + /* PPP Header, etc. */ |
|
short_be(ipcpreq->len) + /* Actual options size + hdr (whole ipcp packet) */ |
|
PPP_FCS_SIZE + /* FCS at the end of the frame */ |
|
1 /* STOP Byte */ |
|
); |
|
} |
|
|
|
static inline uint32_t ipcp_request_options_size(struct pico_device_ppp *ppp) |
|
{ |
|
uint32_t size = 0; |
|
|
|
/* if (ppp->ipcp_ip) */ |
|
size += IPCP_ADDR_LEN; |
|
/* if (ppp->ipcp_dns1) */ |
|
size += IPCP_ADDR_LEN; |
|
/* if (ppp->ipcp_dns2) */ |
|
size += IPCP_ADDR_LEN; |
|
if (ppp->ipcp_nbns1) |
|
size += IPCP_ADDR_LEN; |
|
|
|
if (ppp->ipcp_nbns2) |
|
size += IPCP_ADDR_LEN; |
|
|
|
return size; |
|
} |
|
|
|
static int ipcp_request_add_address(uint8_t *dst, uint8_t tag, uint32_t arg) |
|
{ |
|
uint32_t addr = long_be(arg); |
|
dst[0] = tag; |
|
dst[1] = IPCP_ADDR_LEN; |
|
dst[2] = (uint8_t)((addr & 0xFF000000u) >> 24); |
|
dst[3] = (uint8_t)((addr & 0x00FF0000u) >> 16); |
|
dst[4] = (uint8_t)((addr & 0x0000FF00u) >> 8); |
|
dst[5] = (addr & 0x000000FFu); |
|
return IPCP_ADDR_LEN; |
|
} |
|
|
|
static void ipcp_request_fill(struct pico_device_ppp *ppp, uint8_t *opts) |
|
{ |
|
if (ppp->ipcp_allowed_fields & IPCP_ALLOW_IP) |
|
opts += ipcp_request_add_address(opts, IPCP_OPT_IP, ppp->ipcp_ip); |
|
if (ppp->ipcp_allowed_fields & IPCP_ALLOW_DNS1) |
|
opts += ipcp_request_add_address(opts, IPCP_OPT_DNS1, ppp->ipcp_dns1); |
|
if (ppp->ipcp_allowed_fields & IPCP_ALLOW_DNS2) |
|
opts += ipcp_request_add_address(opts, IPCP_OPT_DNS2, ppp->ipcp_dns2); |
|
if ((ppp->ipcp_allowed_fields & IPCP_ALLOW_NBNS1) && (ppp->ipcp_nbns1)) |
|
opts += ipcp_request_add_address(opts, IPCP_OPT_NBNS1, ppp->ipcp_nbns1); |
|
if ((ppp->ipcp_allowed_fields & IPCP_ALLOW_NBNS2) && (ppp->ipcp_nbns2)) |
|
opts += ipcp_request_add_address(opts, IPCP_OPT_NBNS2, ppp->ipcp_nbns2); |
|
} |
|
|
|
static void ipcp_send_req(struct pico_device_ppp *ppp) |
|
{ |
|
uint8_t ipcp_req[PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_ipcp_hdr) + ipcp_request_options_size(ppp) + PPP_FCS_SIZE + 1]; |
|
uint32_t prefix = PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE; |
|
struct pico_ipcp_hdr *ih = (struct pico_ipcp_hdr *) (ipcp_req + prefix); |
|
uint8_t *p = ipcp_req + prefix + sizeof(struct pico_ipcp_hdr); |
|
uint16_t len = (uint16_t)(ipcp_request_options_size(ppp) + sizeof(struct pico_ipcp_hdr)); |
|
|
|
ih->id = ppp->frame_id++; |
|
ih->code = PICO_CONF_REQ; |
|
ih->len = short_be(len); |
|
ipcp_request_fill(ppp, p); |
|
|
|
ppp_dbg("Sending IPCP CONF REQ, ipcp size = %d\n", len); |
|
pico_ppp_ctl_send(&ppp->dev, PPP_PROTO_IPCP, |
|
ipcp_req, /* Start of PPP packet */ |
|
(uint32_t)(prefix + /* PPP Header, etc. */ |
|
(uint32_t)len + /* IPCP Header + options */ |
|
PPP_FCS_SIZE + /* FCS at the end of the frame */ |
|
1u) /* STOP Byte */ |
|
); |
|
} |
|
|
|
static void ipcp_reject_vj(struct pico_device_ppp *ppp, uint8_t *comp_req) |
|
{ |
|
uint8_t ipcp_req[PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_ipcp_hdr) + IPCP_VJ_LEN + PPP_FCS_SIZE + 1]; |
|
uint32_t prefix = PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE; |
|
struct pico_ipcp_hdr *ih = (struct pico_ipcp_hdr *) (ipcp_req + prefix); |
|
uint8_t *p = ipcp_req + prefix + sizeof(struct pico_ipcp_hdr); |
|
uint32_t i; |
|
|
|
ih->id = ppp->frame_id++; |
|
ih->code = PICO_CONF_REQ; |
|
ih->len = short_be(IPCP_VJ_LEN + sizeof(struct pico_ipcp_hdr)); |
|
for(i = 0; i < IPCP_OPT_VJ; i++) |
|
p[i] = comp_req[i + sizeof(struct pico_ipcp_hdr)]; |
|
ppp_dbg("Sending IPCP CONF REJ VJ\n"); |
|
pico_ppp_ctl_send(&ppp->dev, PPP_PROTO_IPCP, |
|
ipcp_req, /* Start of PPP packet */ |
|
(uint32_t)(prefix + /* PPP Header, etc. */ |
|
sizeof(struct pico_ipcp_hdr) + /* LCP HDR */ |
|
IPCP_VJ_LEN + /* Actual options size */ |
|
PPP_FCS_SIZE + /* FCS at the end of the frame */ |
|
1u) /* STOP Byte */ |
|
); |
|
} |
|
|
|
static void ppp_ipv4_conf(struct pico_device_ppp *ppp) |
|
{ |
|
struct pico_ip4 ip; |
|
struct pico_ip4 nm; |
|
struct pico_ip4 dns1; |
|
struct pico_ip4 dns2; |
|
struct pico_ip4 any = { |
|
0 |
|
}; |
|
ip.addr = ppp->ipcp_ip; |
|
nm.addr = 0xFFFFFF00; |
|
pico_ipv4_link_add(&ppp->dev, ip, nm); |
|
pico_ipv4_route_add(any, any, any, 1, pico_ipv4_link_by_dev(&ppp->dev)); |
|
|
|
dns1.addr = ppp->ipcp_dns1; |
|
dns2.addr = ppp->ipcp_dns2; |
|
pico_dns_client_nameserver(&dns1, PICO_DNS_NS_ADD); |
|
pico_dns_client_nameserver(&dns2, PICO_DNS_NS_ADD); |
|
} |
|
|
|
|
|
static void ipcp_process_in(struct pico_device_ppp *ppp, uint8_t *pkt, uint32_t len) |
|
{ |
|
struct pico_ipcp_hdr *ih = (struct pico_ipcp_hdr *)pkt; |
|
uint8_t *p = pkt + sizeof(struct pico_ipcp_hdr); |
|
int reject = 0; |
|
while (p < pkt + len) { |
|
if (p[0] == IPCP_OPT_VJ) { |
|
reject++; |
|
} |
|
|
|
if (p[0] == IPCP_OPT_IP) { |
|
if (ih->code != PICO_CONF_REJ) |
|
ppp->ipcp_ip = long_be((uint32_t)((p[2] << 24) + (p[3] << 16) + (p[4] << 8) + p[5])); |
|
else { |
|
ppp->ipcp_allowed_fields &= (~IPCP_ALLOW_IP); |
|
ppp->ipcp_ip = 0; |
|
} |
|
} |
|
|
|
if (p[0] == IPCP_OPT_DNS1) { |
|
if (ih->code != PICO_CONF_REJ) |
|
ppp->ipcp_dns1 = long_be((uint32_t)((p[2] << 24) + (p[3] << 16) + (p[4] << 8) + p[5])); |
|
else { |
|
ppp->ipcp_allowed_fields &= (~IPCP_ALLOW_DNS1); |
|
ppp->ipcp_dns1 = 0; |
|
} |
|
} |
|
|
|
if (p[0] == IPCP_OPT_NBNS1) { |
|
if (ih->code != PICO_CONF_REJ) |
|
ppp->ipcp_nbns1 = long_be((uint32_t)((p[2] << 24) + (p[3] << 16) + (p[4] << 8) + p[5])); |
|
else { |
|
ppp->ipcp_allowed_fields &= (~IPCP_ALLOW_NBNS1); |
|
ppp->ipcp_nbns1 = 0; |
|
} |
|
} |
|
|
|
if (p[0] == IPCP_OPT_DNS2) { |
|
if (ih->code != PICO_CONF_REJ) |
|
ppp->ipcp_dns2 = long_be((uint32_t)((p[2] << 24) + (p[3] << 16) + (p[4] << 8) + p[5])); |
|
else { |
|
ppp->ipcp_allowed_fields &= (~IPCP_ALLOW_DNS2); |
|
ppp->ipcp_dns2 = 0; |
|
} |
|
} |
|
|
|
if (p[0] == IPCP_OPT_NBNS2) { |
|
if (ih->code != PICO_CONF_REJ) |
|
ppp->ipcp_nbns2 = long_be((uint32_t)((p[2] << 24) + (p[3] << 16) + (p[4] << 8) + p[5])); |
|
else { |
|
ppp->ipcp_allowed_fields &= (~IPCP_ALLOW_NBNS2); |
|
ppp->ipcp_nbns2 = 0; |
|
} |
|
} |
|
|
|
p += p[1]; |
|
} |
|
if (reject) { |
|
ipcp_reject_vj(ppp, p); |
|
return; |
|
} |
|
|
|
ppp->pkt = pkt; |
|
ppp->len = len; |
|
|
|
switch(ih->code) { |
|
case PICO_CONF_ACK: |
|
ppp_dbg("Received IPCP CONF ACK\n"); |
|
evaluate_ipcp_state(ppp, PPP_IPCP_EVENT_RCA); |
|
break; |
|
case PICO_CONF_REQ: |
|
ppp_dbg("Received IPCP CONF REQ\n"); |
|
evaluate_ipcp_state(ppp, PPP_IPCP_EVENT_RCR_POS); |
|
break; |
|
case PICO_CONF_NAK: |
|
ppp_dbg("Received IPCP CONF NAK\n"); |
|
evaluate_ipcp_state(ppp, PPP_IPCP_EVENT_RCN); |
|
break; |
|
case PICO_CONF_REJ: |
|
ppp_dbg("Received IPCP CONF REJ\n"); |
|
|
|
evaluate_ipcp_state(ppp, PPP_IPCP_EVENT_RCN); |
|
break; |
|
} |
|
} |
|
|
|
static void ipcp6_process_in(struct pico_device_ppp *ppp, uint8_t *pkt, uint32_t len) |
|
{ |
|
IGNORE_PARAMETER(ppp); |
|
IGNORE_PARAMETER(pkt); |
|
IGNORE_PARAMETER(len); |
|
} |
|
|
|
static void ppp_process_packet_payload(struct pico_device_ppp *ppp, uint8_t *pkt, uint32_t len) |
|
{ |
|
if (pkt[0] == 0xc0) { |
|
/* Link control packet */ |
|
if (pkt[1] == 0x21) { |
|
/* LCP */ |
|
lcp_process_in(ppp, pkt + 2, len - 2); |
|
} |
|
|
|
if (pkt[1] == 0x23) { |
|
/* PAP */ |
|
pap_process_in(ppp, pkt + 2, len - 2); |
|
} |
|
|
|
return; |
|
} |
|
|
|
if ((pkt[0] == 0xc2) && (pkt[1] == 0x23)) { |
|
/* CHAP */ |
|
chap_process_in(ppp, pkt + 2, len - 2); |
|
return; |
|
} |
|
|
|
if (pkt[0] == 0x80) { |
|
/* IP assignment (IPCP/IPCP6) */ |
|
if (pkt[1] == 0x21) { |
|
/* IPCP */ |
|
ipcp_process_in(ppp, pkt + 2, len - 2); |
|
} |
|
|
|
if (pkt[1] == 0x57) { |
|
/* IPCP6 */ |
|
ipcp6_process_in(ppp, pkt + 2, len - 2); |
|
} |
|
|
|
return; |
|
} |
|
|
|
if (pkt[0] == 0x00) { |
|
/* Uncompressed protocol: leading zero. */ |
|
pkt++; |
|
len--; |
|
} |
|
|
|
if ((pkt[0] == 0x21) || (pkt[0] == 0x57)) { |
|
/* IPv4 /v6 Data */ |
|
pico_stack_recv(&ppp->dev, pkt + 1, len - 1); |
|
return; |
|
} |
|
|
|
ppp_dbg("PPP: Unrecognized protocol %02x%02x\n", pkt[0], pkt[1]); |
|
} |
|
|
|
static void ppp_process_packet(struct pico_device_ppp *ppp, uint8_t *pkt, uint32_t len) |
|
{ |
|
/* Verify incoming FCS */ |
|
if (ppp_fcs_verify(pkt, len) != 0) |
|
return; |
|
|
|
/* Remove trailing FCS */ |
|
len -= 2; |
|
|
|
/* Remove ADDR/CTRL, then process */ |
|
if ((pkt[0] == PPPF_ADDR) && (pkt[1] == PPPF_CTRL)) { |
|
pkt += 2; |
|
len -= 2; |
|
} |
|
|
|
ppp_process_packet_payload(ppp, pkt, len); |
|
|
|
} |
|
|
|
static void ppp_recv_data(struct pico_device_ppp *ppp, void *data, uint32_t len) |
|
{ |
|
uint8_t *pkt = (uint8_t *)data; |
|
|
|
#ifdef PPP_DEBUG |
|
uint32_t idx; |
|
if (len > 0) { |
|
ppp_dbg("PPP <<<<< "); |
|
for(idx = 0; idx < len; idx++) { |
|
ppp_dbg(" %02x", ((uint8_t *)data)[idx]); |
|
} |
|
ppp_dbg("\n"); |
|
} |
|
#endif |
|
|
|
ppp_process_packet(ppp, pkt, len); |
|
} |
|
|
|
static void lcp_this_layer_up(struct pico_device_ppp *ppp) |
|
{ |
|
ppp_dbg("PPP: LCP up.\n"); |
|
|
|
switch (ppp->auth) { |
|
case 0x0000: |
|
evaluate_auth_state(ppp, PPP_AUTH_EVENT_UP_NONE); |
|
break; |
|
case 0xc023: |
|
evaluate_auth_state(ppp, PPP_AUTH_EVENT_UP_PAP); |
|
break; |
|
case 0xc223: |
|
evaluate_auth_state(ppp, PPP_AUTH_EVENT_UP_CHAP); |
|
break; |
|
default: |
|
ppp_dbg("PPP: Unknown authentication protocol.\n"); |
|
break; |
|
} |
|
} |
|
|
|
static void lcp_this_layer_down(struct pico_device_ppp *ppp) |
|
{ |
|
ppp_dbg("PPP: LCP down.\n"); |
|
evaluate_auth_state(ppp, PPP_AUTH_EVENT_DOWN); |
|
} |
|
|
|
static void lcp_this_layer_started(struct pico_device_ppp *ppp) |
|
{ |
|
ppp_dbg("PPP: LCP started.\n"); |
|
evaluate_modem_state(ppp, PPP_MODEM_EVENT_START); |
|
} |
|
|
|
static void lcp_this_layer_finished(struct pico_device_ppp *ppp) |
|
{ |
|
ppp_dbg("PPP: LCP finished.\n"); |
|
evaluate_modem_state(ppp, PPP_MODEM_EVENT_STOP); |
|
} |
|
|
|
static void lcp_initialize_restart_count(struct pico_device_ppp *ppp) |
|
{ |
|
lcp_timer_start(ppp, PPP_TIMER_ON_LCPREQ); |
|
} |
|
|
|
static void lcp_send_code_reject(struct pico_device_ppp *ppp) |
|
{ |
|
IGNORE_PARAMETER(ppp); |
|
} |
|
|
|
static void lcp_send_echo_reply(struct pico_device_ppp *ppp) |
|
{ |
|
uint8_t reply[ppp->len + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_lcp_hdr) + PPP_FCS_SIZE + 1]; |
|
struct pico_lcp_hdr *reply_hdr = (struct pico_lcp_hdr *) (reply + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE); |
|
struct pico_lcp_hdr *lcpreq = (struct pico_lcp_hdr *)ppp->pkt; |
|
memcpy(reply + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE, ppp->pkt, ppp->len); |
|
reply_hdr->code = PICO_CONF_ECHO_REP; |
|
reply_hdr->id = lcpreq->id; |
|
reply_hdr->len = lcpreq->len; |
|
ppp_dbg("Sending LCP ECHO REPLY\n"); |
|
pico_ppp_ctl_send(&ppp->dev, PPP_PROTO_LCP, reply, |
|
PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + /* PPP Header, etc. */ |
|
short_be(lcpreq->len) + /* Actual options size + hdr (whole lcp packet) */ |
|
PPP_FCS_SIZE + /* FCS at the end of the frame */ |
|
1 /* STOP Byte */ |
|
); |
|
} |
|
|
|
static const struct pico_ppp_fsm ppp_lcp_fsm[PPP_LCP_STATE_MAX][PPP_LCP_EVENT_MAX] = { |
|
[PPP_LCP_STATE_INITIAL] = { |
|
[PPP_LCP_EVENT_UP] = { PPP_LCP_STATE_CLOSED, {} }, |
|
[PPP_LCP_EVENT_DOWN] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_OPEN] = { PPP_LCP_STATE_STARTING, { lcp_this_layer_started } }, |
|
[PPP_LCP_EVENT_CLOSE] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_TO_POS] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_TO_NEG] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_RCR_POS] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_RCR_NEG] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_RCA] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_RCN] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_RTR] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_RTA] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_RUC] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_RXJ_POS] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_RXJ_NEG] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_RXR] = { PPP_LCP_STATE_INITIAL, {} } |
|
}, |
|
[PPP_LCP_STATE_STARTING] = { |
|
[PPP_LCP_EVENT_UP] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_configure_request } }, |
|
[PPP_LCP_EVENT_DOWN] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_OPEN] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_CLOSE] = { PPP_LCP_STATE_INITIAL, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_TO_POS] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_TO_NEG] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_RCR_POS] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_RCR_NEG] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_RCA] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_RCN] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_RTR] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_RTA] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_RUC] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_RXJ_POS] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_RXJ_NEG] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_RXR] = { PPP_LCP_STATE_STARTING, {} } |
|
}, |
|
[PPP_LCP_STATE_CLOSED] = { |
|
[PPP_LCP_EVENT_UP] = { PPP_LCP_STATE_CLOSED, {} }, |
|
[PPP_LCP_EVENT_DOWN] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_OPEN] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_configure_request} }, |
|
[PPP_LCP_EVENT_CLOSE] = { PPP_LCP_STATE_CLOSED, {} }, |
|
[PPP_LCP_EVENT_TO_POS] = { PPP_LCP_STATE_CLOSED, {} }, |
|
[PPP_LCP_EVENT_TO_NEG] = { PPP_LCP_STATE_CLOSED, {} }, |
|
[PPP_LCP_EVENT_RCR_POS] = { PPP_LCP_STATE_CLOSED, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RCR_NEG] = { PPP_LCP_STATE_CLOSED, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RCA] = { PPP_LCP_STATE_CLOSED, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RCN] = { PPP_LCP_STATE_CLOSED, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RTR] = { PPP_LCP_STATE_CLOSED, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RTA] = { PPP_LCP_STATE_CLOSED, {} }, |
|
[PPP_LCP_EVENT_RUC] = { PPP_LCP_STATE_CLOSED, { lcp_send_code_reject } }, |
|
[PPP_LCP_EVENT_RXJ_POS] = { PPP_LCP_STATE_CLOSED, {} }, |
|
[PPP_LCP_EVENT_RXJ_NEG] = { PPP_LCP_STATE_CLOSED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RXR] = { PPP_LCP_STATE_CLOSED, {} } |
|
}, |
|
[PPP_LCP_STATE_STOPPED] = { |
|
[PPP_LCP_EVENT_UP] = { PPP_LCP_STATE_STOPPED, {} }, |
|
[PPP_LCP_EVENT_DOWN] = { PPP_LCP_STATE_STARTING, { lcp_this_layer_started } }, |
|
[PPP_LCP_EVENT_OPEN] = { PPP_LCP_STATE_STOPPED, {}}, |
|
[PPP_LCP_EVENT_CLOSE] = { PPP_LCP_STATE_CLOSED, {}}, |
|
[PPP_LCP_EVENT_TO_POS] = { PPP_LCP_STATE_STOPPED, {} }, |
|
[PPP_LCP_EVENT_TO_NEG] = { PPP_LCP_STATE_STOPPED, {} }, |
|
[PPP_LCP_EVENT_RCR_POS] = { PPP_LCP_STATE_ACK_SENT, |
|
{ lcp_send_configure_request, lcp_send_configure_ack}}, |
|
[PPP_LCP_EVENT_RCR_NEG] = { PPP_LCP_STATE_REQ_SENT, |
|
{ lcp_send_configure_request, lcp_send_configure_nack}}, |
|
[PPP_LCP_EVENT_RCA] = { PPP_LCP_STATE_STOPPED, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RCN] = { PPP_LCP_STATE_STOPPED, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RTR] = { PPP_LCP_STATE_STOPPED, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RTA] = { PPP_LCP_STATE_STOPPED, {} }, |
|
[PPP_LCP_EVENT_RUC] = { PPP_LCP_STATE_STOPPED, { lcp_send_code_reject } }, |
|
[PPP_LCP_EVENT_RXJ_POS] = { PPP_LCP_STATE_STOPPED, {} }, |
|
[PPP_LCP_EVENT_RXJ_NEG] = { PPP_LCP_STATE_STOPPED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RXR] = { PPP_LCP_STATE_STOPPED, {} } |
|
}, |
|
[PPP_LCP_STATE_CLOSING] = { |
|
[PPP_LCP_EVENT_UP] = { PPP_LCP_STATE_CLOSING, {} }, |
|
[PPP_LCP_EVENT_DOWN] = { PPP_LCP_STATE_INITIAL, {} }, |
|
[PPP_LCP_EVENT_OPEN] = { PPP_LCP_STATE_STOPPING, {} }, |
|
[PPP_LCP_EVENT_CLOSE] = { PPP_LCP_STATE_CLOSING, {} }, |
|
[PPP_LCP_EVENT_TO_POS] = { PPP_LCP_STATE_CLOSING, { lcp_send_terminate_request } }, |
|
[PPP_LCP_EVENT_TO_NEG] = { PPP_LCP_STATE_CLOSED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RCR_POS] = { PPP_LCP_STATE_CLOSING, {} }, |
|
[PPP_LCP_EVENT_RCR_NEG] = { PPP_LCP_STATE_CLOSING, {} }, |
|
[PPP_LCP_EVENT_RCA] = { PPP_LCP_STATE_CLOSING, {} }, |
|
[PPP_LCP_EVENT_RCN] = { PPP_LCP_STATE_CLOSING, {} }, |
|
[PPP_LCP_EVENT_RTR] = { PPP_LCP_STATE_CLOSING, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RTA] = { PPP_LCP_STATE_CLOSED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RUC] = { PPP_LCP_STATE_CLOSING, { lcp_send_code_reject } }, |
|
[PPP_LCP_EVENT_RXJ_POS] = { PPP_LCP_STATE_CLOSING, {} }, |
|
[PPP_LCP_EVENT_RXJ_NEG] = { PPP_LCP_STATE_CLOSED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RXR] = { PPP_LCP_STATE_CLOSING, {} } |
|
}, |
|
[PPP_LCP_STATE_STOPPING] = { |
|
[PPP_LCP_EVENT_UP] = { PPP_LCP_STATE_STOPPING, {} }, |
|
[PPP_LCP_EVENT_DOWN] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_OPEN] = { PPP_LCP_STATE_STOPPING, {} }, |
|
[PPP_LCP_EVENT_CLOSE] = { PPP_LCP_STATE_CLOSING, {} }, |
|
[PPP_LCP_EVENT_TO_POS] = { PPP_LCP_STATE_STOPPING, { lcp_send_terminate_request } }, |
|
[PPP_LCP_EVENT_TO_NEG] = { PPP_LCP_STATE_STOPPED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RCR_POS] = { PPP_LCP_STATE_STOPPING, {} }, |
|
[PPP_LCP_EVENT_RCR_NEG] = { PPP_LCP_STATE_STOPPING, {} }, |
|
[PPP_LCP_EVENT_RCA] = { PPP_LCP_STATE_STOPPING, {} }, |
|
[PPP_LCP_EVENT_RCN] = { PPP_LCP_STATE_STOPPING, {} }, |
|
[PPP_LCP_EVENT_RTR] = { PPP_LCP_STATE_STOPPING, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RTA] = { PPP_LCP_STATE_STOPPED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RUC] = { PPP_LCP_STATE_STOPPING, { lcp_send_code_reject } }, |
|
[PPP_LCP_EVENT_RXJ_POS] = { PPP_LCP_STATE_STOPPING, {} }, |
|
[PPP_LCP_EVENT_RXJ_NEG] = { PPP_LCP_STATE_STOPPED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RXR] = { PPP_LCP_STATE_STOPPING, {} } |
|
}, |
|
[PPP_LCP_STATE_REQ_SENT] = { |
|
[PPP_LCP_EVENT_UP] = { PPP_LCP_STATE_REQ_SENT, {} }, |
|
[PPP_LCP_EVENT_DOWN] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_OPEN] = { PPP_LCP_STATE_REQ_SENT, {} }, |
|
[PPP_LCP_EVENT_CLOSE] = { PPP_LCP_STATE_CLOSING, { lcp_send_terminate_request } }, |
|
[PPP_LCP_EVENT_TO_POS] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_configure_request } }, |
|
[PPP_LCP_EVENT_TO_NEG] = { PPP_LCP_STATE_STOPPED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RCR_POS] = { PPP_LCP_STATE_ACK_SENT, { lcp_send_configure_ack } }, |
|
[PPP_LCP_EVENT_RCR_NEG] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_configure_nack } }, |
|
[PPP_LCP_EVENT_RCA] = { PPP_LCP_STATE_ACK_RCVD, { lcp_initialize_restart_count } }, |
|
[PPP_LCP_EVENT_RCN] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_configure_request} }, |
|
[PPP_LCP_EVENT_RTR] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RTA] = { PPP_LCP_STATE_REQ_SENT, {} }, |
|
[PPP_LCP_EVENT_RUC] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_code_reject } }, |
|
[PPP_LCP_EVENT_RXJ_POS] = { PPP_LCP_STATE_REQ_SENT, {} }, |
|
[PPP_LCP_EVENT_RXJ_NEG] = { PPP_LCP_STATE_STOPPED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RXR] = { PPP_LCP_STATE_REQ_SENT, {} } |
|
}, |
|
[PPP_LCP_STATE_ACK_RCVD] = { |
|
[PPP_LCP_EVENT_UP] = { PPP_LCP_STATE_ACK_RCVD, {} }, |
|
[PPP_LCP_EVENT_DOWN] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_OPEN] = { PPP_LCP_STATE_ACK_RCVD, {} }, |
|
[PPP_LCP_EVENT_CLOSE] = { PPP_LCP_STATE_CLOSING, { lcp_send_terminate_request} }, |
|
[PPP_LCP_EVENT_TO_POS] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_configure_request } }, |
|
[PPP_LCP_EVENT_TO_NEG] = { PPP_LCP_STATE_STOPPED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RCR_POS] = { PPP_LCP_STATE_OPENED, { lcp_send_configure_ack, lcp_this_layer_up} }, |
|
[PPP_LCP_EVENT_RCR_NEG] = { PPP_LCP_STATE_ACK_RCVD, { lcp_send_configure_nack } }, |
|
[PPP_LCP_EVENT_RCA] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_configure_request } }, |
|
[PPP_LCP_EVENT_RCN] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_configure_request } }, |
|
[PPP_LCP_EVENT_RTR] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RTA] = { PPP_LCP_STATE_REQ_SENT, {} }, |
|
[PPP_LCP_EVENT_RUC] = { PPP_LCP_STATE_ACK_RCVD, { lcp_send_code_reject } }, |
|
[PPP_LCP_EVENT_RXJ_POS] = { PPP_LCP_STATE_REQ_SENT, {} }, |
|
[PPP_LCP_EVENT_RXJ_NEG] = { PPP_LCP_STATE_STOPPED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RXR] = { PPP_LCP_STATE_ACK_RCVD, {} } |
|
}, |
|
[PPP_LCP_STATE_ACK_SENT] = { |
|
[PPP_LCP_EVENT_UP] = { PPP_LCP_STATE_ACK_SENT, {} }, |
|
[PPP_LCP_EVENT_DOWN] = { PPP_LCP_STATE_STARTING, {} }, |
|
[PPP_LCP_EVENT_OPEN] = { PPP_LCP_STATE_ACK_SENT, {} }, |
|
[PPP_LCP_EVENT_CLOSE] = { PPP_LCP_STATE_CLOSING, { lcp_send_terminate_request} }, |
|
[PPP_LCP_EVENT_TO_POS] = { PPP_LCP_STATE_ACK_SENT, { lcp_send_configure_request } }, |
|
[PPP_LCP_EVENT_TO_NEG] = { PPP_LCP_STATE_STOPPED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RCR_POS] = { PPP_LCP_STATE_ACK_SENT, { lcp_send_configure_ack } }, |
|
[PPP_LCP_EVENT_RCR_NEG] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_configure_nack } }, |
|
[PPP_LCP_EVENT_RCA] = { PPP_LCP_STATE_OPENED, { lcp_this_layer_up} }, |
|
[PPP_LCP_EVENT_RCN] = { PPP_LCP_STATE_ACK_SENT, { lcp_send_configure_request} }, |
|
[PPP_LCP_EVENT_RTR] = { PPP_LCP_STATE_REQ_SENT, { lcp_send_terminate_ack } }, |
|
[PPP_LCP_EVENT_RTA] = { PPP_LCP_STATE_ACK_SENT, {} }, |
|
[PPP_LCP_EVENT_RUC] = { PPP_LCP_STATE_ACK_SENT, { lcp_send_code_reject } }, |
|
[PPP_LCP_EVENT_RXJ_POS] = { PPP_LCP_STATE_ACK_SENT, {} }, |
|
[PPP_LCP_EVENT_RXJ_NEG] = { PPP_LCP_STATE_STOPPED, { lcp_this_layer_finished } }, |
|
[PPP_LCP_EVENT_RXR] = { PPP_LCP_STATE_ACK_SENT, {} } |
|
}, |
|
[PPP_LCP_STATE_OPENED] = { |
|
[PPP_LCP_EVENT_UP] = { PPP_LCP_STATE_OPENED, {} }, |
|
[PPP_LCP_EVENT_DOWN] = { PPP_LCP_STATE_STARTING, {lcp_this_layer_down } }, |
|
[PPP_LCP_EVENT_OPEN] = { PPP_LCP_STATE_OPENED, {} }, |
|
[PPP_LCP_EVENT_CLOSE] = { PPP_LCP_STATE_CLOSING, |
|
{ lcp_this_layer_down, lcp_send_terminate_request }}, |
|
[PPP_LCP_EVENT_TO_POS] = { PPP_LCP_STATE_OPENED, {} }, |
|
[PPP_LCP_EVENT_TO_NEG] = { PPP_LCP_STATE_OPENED, {} }, |
|
[PPP_LCP_EVENT_RCR_POS] = { PPP_LCP_STATE_ACK_SENT, |
|
{ lcp_this_layer_down, lcp_send_terminate_request, lcp_send_configure_ack }}, |
|
[PPP_LCP_EVENT_RCR_NEG] = { PPP_LCP_STATE_REQ_SENT, |
|
{ lcp_this_layer_down, lcp_send_configure_request, lcp_send_configure_nack }}, |
|
[PPP_LCP_EVENT_RCA] = { PPP_LCP_STATE_REQ_SENT, { lcp_this_layer_down, lcp_send_terminate_request } }, |
|
[PPP_LCP_EVENT_RCN] = { PPP_LCP_STATE_REQ_SENT, { lcp_this_layer_down, lcp_send_terminate_request } }, |
|
[PPP_LCP_EVENT_RTR] = { PPP_LCP_STATE_STOPPING, { lcp_this_layer_down, lcp_zero_restart_count, lcp_send_terminate_ack} }, |
|
[PPP_LCP_EVENT_RTA] = { PPP_LCP_STATE_REQ_SENT, { lcp_this_layer_down, lcp_send_terminate_request} }, |
|
[PPP_LCP_EVENT_RUC] = { PPP_LCP_STATE_OPENED, { lcp_send_code_reject } }, |
|
[PPP_LCP_EVENT_RXJ_POS] = { PPP_LCP_STATE_OPENED, { } }, |
|
[PPP_LCP_EVENT_RXJ_NEG] = { PPP_LCP_STATE_STOPPING, |
|
{lcp_this_layer_down, lcp_send_terminate_request}}, |
|
[PPP_LCP_EVENT_RXR] = { PPP_LCP_STATE_OPENED, { lcp_send_echo_reply} } |
|
} |
|
}; |
|
|
|
static void evaluate_lcp_state(struct pico_device_ppp *ppp, enum ppp_lcp_event event) |
|
{ |
|
const struct pico_ppp_fsm *fsm, *next_fsm_to; |
|
int i; |
|
if (!ppp) |
|
return; |
|
|
|
if (mock_lcp_state) { |
|
mock_lcp_state(ppp, event); |
|
return; |
|
} |
|
|
|
fsm = &ppp_lcp_fsm[ppp->lcp_state][event]; |
|
ppp->lcp_state = (enum ppp_lcp_state)fsm->next_state; |
|
/* RFC1661: The states in which the Restart timer is running are identifiable by |
|
* the presence of TO events. |
|
*/ |
|
next_fsm_to = &ppp_lcp_fsm[ppp->lcp_state][PPP_LCP_EVENT_TO_POS]; |
|
if (!next_fsm_to->event_handler[0]) { |
|
/* The Restart timer is stopped when transitioning |
|
* from any state where the timer is running to a state where the timer |
|
* is not running. |
|
*/ |
|
lcp_timer_stop(ppp, PPP_TIMER_ON_LCPREQ); |
|
lcp_timer_stop(ppp, PPP_TIMER_ON_LCPTERM); |
|
} |
|
|
|
for (i = 0; i < PPP_FSM_MAX_ACTIONS; i++) { |
|
if (fsm->event_handler[i]) |
|
fsm->event_handler[i](ppp); |
|
} |
|
} |
|
|
|
static void auth(struct pico_device_ppp *ppp) |
|
{ |
|
ppp_dbg("PPP: Authenticated.\n"); |
|
ppp->ipcp_allowed_fields = 0xFFFF; |
|
evaluate_ipcp_state(ppp, PPP_IPCP_EVENT_UP); |
|
} |
|
|
|
static void deauth(struct pico_device_ppp *ppp) |
|
{ |
|
ppp_dbg("PPP: De-authenticated.\n"); |
|
evaluate_ipcp_state(ppp, PPP_IPCP_EVENT_DOWN); |
|
} |
|
|
|
static void auth_abort(struct pico_device_ppp *ppp) |
|
{ |
|
ppp_dbg("PPP: Authentication failed!\n"); |
|
ppp->timer_on = (uint8_t) (ppp->timer_on & (~PPP_TIMER_ON_AUTH)); |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_CLOSE); |
|
|
|
} |
|
|
|
static void auth_req(struct pico_device_ppp *ppp) |
|
{ |
|
uint16_t ppp_usr_len = 0; |
|
uint16_t ppp_pwd_len = 0; |
|
uint8_t *req = NULL, *p; |
|
struct pico_pap_hdr *hdr; |
|
uint16_t pap_len = 0; |
|
uint8_t field_len = 0; |
|
ppp_usr_len = (uint16_t)strlen(ppp->username); |
|
ppp_pwd_len = (uint16_t)strlen(ppp->password); |
|
|
|
pap_len = (uint16_t)(sizeof(struct pico_pap_hdr) + 1u + 1u + ppp_usr_len + ppp_pwd_len); |
|
|
|
req = PICO_ZALLOC(PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + pap_len + PPP_FCS_SIZE + 1); |
|
if (!req) |
|
return; |
|
|
|
hdr = (struct pico_pap_hdr *) (req + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE); |
|
|
|
hdr->code = PAP_AUTH_REQ; |
|
hdr->id = ppp->frame_id++; |
|
hdr->len = short_be(pap_len); |
|
|
|
p = req + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_pap_hdr); |
|
|
|
/* Populate authentication domain */ |
|
field_len = (uint8_t)(ppp_usr_len & 0xFF); |
|
*p = field_len; |
|
++p; |
|
if (ppp_usr_len > 0) { |
|
memcpy(p, ppp->username, ppp_usr_len); |
|
p += ppp_usr_len; |
|
} |
|
|
|
/* Populate authentication password */ |
|
field_len = (uint8_t)(ppp_pwd_len & 0xFF); |
|
*p = field_len; |
|
++p; |
|
if (ppp_pwd_len > 0) { |
|
memcpy(p, ppp->password, ppp_pwd_len); |
|
p += ppp_pwd_len; |
|
} |
|
ppp_dbg("PAP: Sending authentication request.\n"); |
|
pico_ppp_ctl_send(&ppp->dev, PPP_PROTO_PAP, |
|
req, /* Start of PPP packet */ |
|
(uint32_t)( |
|
PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + /* PPP Header, etc. */ |
|
pap_len + /* Authentication packet len */ |
|
PPP_FCS_SIZE + /* FCS */ |
|
1) /* STOP Byte */ |
|
); |
|
PICO_FREE(req); |
|
} |
|
|
|
static void auth_rsp(struct pico_device_ppp *ppp) |
|
{ |
|
struct pico_chap_hdr *ch = (struct pico_chap_hdr *)ppp->pkt; |
|
uint8_t resp[PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_chap_hdr) + CHAP_MD5_SIZE + PPP_FCS_SIZE + 2]; |
|
struct pico_chap_hdr *rh = (struct pico_chap_hdr *) (resp + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE); |
|
uint8_t *md5resp = resp + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_chap_hdr) + 1; |
|
uint8_t *md5resp_len = resp + PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + sizeof(struct pico_chap_hdr); |
|
uint8_t *challenge; |
|
uint32_t i = 0, pwdlen; |
|
uint8_t *recvd_challenge_len = ppp->pkt + sizeof(struct pico_chap_hdr); |
|
uint8_t *recvd_challenge = recvd_challenge_len + 1; |
|
size_t challenge_size = CHALLENGE_SIZE(ppp, ch); |
|
|
|
challenge = PICO_ZALLOC(challenge_size); |
|
|
|
if (!challenge) |
|
return; |
|
|
|
|
|
pwdlen = (uint32_t)strlen(ppp->password); |
|
challenge[i++] = ch->id; |
|
memcpy(challenge + i, ppp->password, pwdlen); |
|
i += pwdlen; |
|
memcpy(challenge + i, recvd_challenge, *recvd_challenge_len); |
|
i += *recvd_challenge_len; |
|
pico_md5sum(md5resp, challenge, i); |
|
PICO_FREE(challenge); |
|
rh->id = ch->id; |
|
rh->code = CHAP_RESPONSE; |
|
rh->len = short_be(CHAP_MD5_SIZE + sizeof(struct pico_chap_hdr) + 1); |
|
*md5resp_len = CHAP_MD5_SIZE; |
|
ppp_dbg("Sending CHAP RESPONSE, \n"); |
|
pico_ppp_ctl_send(&ppp->dev, PPP_PROTO_CHAP, |
|
resp, /* Start of PPP packet */ |
|
(uint32_t)( |
|
PPP_HDR_SIZE + PPP_PROTO_SLOT_SIZE + /* PPP Header, etc. */ |
|
sizeof(struct pico_chap_hdr) + /* CHAP HDR */ |
|
1 + /* Value length */ |
|
CHAP_MD5_SIZE + /* Actual payload size */ |
|
PPP_FCS_SIZE + /* FCS at the end of the frame */ |
|
1) /* STOP Byte */ |
|
); |
|
} |
|
|
|
static void auth_start_timer(struct pico_device_ppp *ppp) |
|
{ |
|
ppp->timer_on = ppp->timer_on | PPP_TIMER_ON_AUTH; |
|
ppp->timer_val = PICO_PPP_DEFAULT_TIMER; |
|
} |
|
|
|
static const struct pico_ppp_fsm ppp_auth_fsm[PPP_AUTH_STATE_MAX][PPP_AUTH_EVENT_MAX] = { |
|
[PPP_AUTH_STATE_INITIAL] = { |
|
[PPP_AUTH_EVENT_UP_NONE] = { PPP_AUTH_STATE_AUTHENTICATED, {auth} }, |
|
[PPP_AUTH_EVENT_UP_PAP] = { PPP_AUTH_STATE_REQ_SENT, {auth_req, auth_start_timer} }, |
|
[PPP_AUTH_EVENT_UP_CHAP] = { PPP_AUTH_STATE_STARTING, {} }, |
|
[PPP_AUTH_EVENT_DOWN] = { PPP_AUTH_STATE_INITIAL, {} }, |
|
[PPP_AUTH_EVENT_RAC] = { PPP_AUTH_STATE_INITIAL, {} }, |
|
[PPP_AUTH_EVENT_RAA] = { PPP_AUTH_STATE_INITIAL, {} }, |
|
[PPP_AUTH_EVENT_RAN] = { PPP_AUTH_STATE_INITIAL, {auth_abort} }, |
|
[PPP_AUTH_EVENT_TO] = { PPP_AUTH_STATE_INITIAL, {} } |
|
}, |
|
[PPP_AUTH_STATE_STARTING] = { |
|
[PPP_AUTH_EVENT_UP_NONE] = { PPP_AUTH_STATE_STARTING, {} }, |
|
[PPP_AUTH_EVENT_UP_PAP] = { PPP_AUTH_STATE_STARTING, {} }, |
|
[PPP_AUTH_EVENT_UP_CHAP] = { PPP_AUTH_STATE_STARTING, {} }, |
|
[PPP_AUTH_EVENT_DOWN] = { PPP_AUTH_STATE_INITIAL, {deauth} }, |
|
[PPP_AUTH_EVENT_RAC] = { PPP_AUTH_STATE_RSP_SENT, {auth_rsp, auth_start_timer} }, |
|
[PPP_AUTH_EVENT_RAA] = { PPP_AUTH_STATE_STARTING, {auth_start_timer} }, |
|
[PPP_AUTH_EVENT_RAN] = { PPP_AUTH_STATE_STARTING, {auth_abort} }, |
|
[PPP_AUTH_EVENT_TO] = { PPP_AUTH_STATE_INITIAL, {auth_req, auth_start_timer} } |
|
}, |
|
[PPP_AUTH_STATE_RSP_SENT] = { |
|
[PPP_AUTH_EVENT_UP_NONE] = { PPP_AUTH_STATE_RSP_SENT, {} }, |
|
[PPP_AUTH_EVENT_UP_PAP] = { PPP_AUTH_STATE_RSP_SENT, {} }, |
|
[PPP_AUTH_EVENT_UP_CHAP] = { PPP_AUTH_STATE_RSP_SENT, {} }, |
|
[PPP_AUTH_EVENT_DOWN] = { PPP_AUTH_STATE_INITIAL, {deauth} }, |
|
[PPP_AUTH_EVENT_RAC] = { PPP_AUTH_STATE_RSP_SENT, {auth_rsp, auth_start_timer} }, |
|
[PPP_AUTH_EVENT_RAA] = { PPP_AUTH_STATE_AUTHENTICATED, {auth} }, |
|
[PPP_AUTH_EVENT_RAN] = { PPP_AUTH_STATE_STARTING, {auth_abort} }, |
|
[PPP_AUTH_EVENT_TO] = { PPP_AUTH_STATE_STARTING, {auth_start_timer} } |
|
}, |
|
[PPP_AUTH_STATE_REQ_SENT] = { |
|
[PPP_AUTH_EVENT_UP_NONE] = { PPP_AUTH_STATE_REQ_SENT, {} }, |
|
[PPP_AUTH_EVENT_UP_PAP] = { PPP_AUTH_STATE_REQ_SENT, {} }, |
|
[PPP_AUTH_EVENT_UP_CHAP] = { PPP_AUTH_STATE_REQ_SENT, {} }, |
|
[PPP_AUTH_EVENT_DOWN] = { PPP_AUTH_STATE_INITIAL, {deauth} }, |
|
[PPP_AUTH_EVENT_RAC] = { PPP_AUTH_STATE_REQ_SENT, {} }, |
|
[PPP_AUTH_EVENT_RAA] = { PPP_AUTH_STATE_AUTHENTICATED, {auth} }, |
|
[PPP_AUTH_EVENT_RAN] = { PPP_AUTH_STATE_REQ_SENT, {auth_abort} }, |
|
[PPP_AUTH_EVENT_TO] = { PPP_AUTH_STATE_REQ_SENT, {auth_req, auth_start_timer} } |
|
}, |
|
[PPP_AUTH_STATE_AUTHENTICATED] = { |
|
[PPP_AUTH_EVENT_UP_NONE] = { PPP_AUTH_STATE_AUTHENTICATED, {} }, |
|
[PPP_AUTH_EVENT_UP_PAP] = { PPP_AUTH_STATE_AUTHENTICATED, {} }, |
|
[PPP_AUTH_EVENT_UP_CHAP] = { PPP_AUTH_STATE_AUTHENTICATED, {} }, |
|
[PPP_AUTH_EVENT_DOWN] = { PPP_AUTH_STATE_INITIAL, {deauth} }, |
|
[PPP_AUTH_EVENT_RAC] = { PPP_AUTH_STATE_RSP_SENT, {auth_rsp} }, |
|
[PPP_AUTH_EVENT_RAA] = { PPP_AUTH_STATE_AUTHENTICATED, {} }, |
|
[PPP_AUTH_EVENT_RAN] = { PPP_AUTH_STATE_AUTHENTICATED, {} }, |
|
[PPP_AUTH_EVENT_TO] = { PPP_AUTH_STATE_AUTHENTICATED, {} }, |
|
} |
|
}; |
|
|
|
static void evaluate_auth_state(struct pico_device_ppp *ppp, enum ppp_auth_event event) |
|
{ |
|
const struct pico_ppp_fsm *fsm; |
|
int i; |
|
if (mock_auth_state) { |
|
mock_auth_state(ppp, event); |
|
return; |
|
} |
|
|
|
fsm = &ppp_auth_fsm[ppp->auth_state][event]; |
|
|
|
ppp->auth_state = (enum ppp_auth_state)fsm->next_state; |
|
for (i = 0; i < PPP_FSM_MAX_ACTIONS; i++) { |
|
if (fsm->event_handler[i]) |
|
fsm->event_handler[i](ppp); |
|
} |
|
} |
|
|
|
static void ipcp_send_nack(struct pico_device_ppp *ppp) |
|
{ |
|
IGNORE_PARAMETER(ppp); |
|
} |
|
|
|
static void ipcp_bring_up(struct pico_device_ppp *ppp) |
|
{ |
|
ppp_dbg("PPP: IPCP up.\n"); |
|
|
|
if (ppp->ipcp_ip) { |
|
char my_ip[16], my_dns[16]; |
|
pico_ipv4_to_string(my_ip, ppp->ipcp_ip); |
|
ppp_dbg("Received IP config %s\n", my_ip); |
|
pico_ipv4_to_string(my_dns, ppp->ipcp_dns1); |
|
ppp_dbg("Received DNS: %s\n", my_dns); |
|
ppp_ipv4_conf(ppp); |
|
} |
|
} |
|
|
|
static void ipcp_bring_down(struct pico_device_ppp *ppp) |
|
{ |
|
IGNORE_PARAMETER(ppp); |
|
|
|
ppp_dbg("PPP: IPCP down.\n"); |
|
} |
|
|
|
static void ipcp_start_timer(struct pico_device_ppp *ppp) |
|
{ |
|
ppp->timer_on = ppp->timer_on | PPP_TIMER_ON_IPCP; |
|
ppp->timer_val = PICO_PPP_DEFAULT_TIMER * PICO_PPP_DEFAULT_MAX_FAILURE; |
|
} |
|
|
|
static const struct pico_ppp_fsm ppp_ipcp_fsm[PPP_IPCP_STATE_MAX][PPP_IPCP_EVENT_MAX] = { |
|
[PPP_IPCP_STATE_INITIAL] = { |
|
[PPP_IPCP_EVENT_UP] = { PPP_IPCP_STATE_REQ_SENT, {ipcp_send_req, ipcp_start_timer} }, |
|
[PPP_IPCP_EVENT_DOWN] = { PPP_IPCP_STATE_INITIAL, {} }, |
|
[PPP_IPCP_EVENT_RCR_POS] = { PPP_IPCP_STATE_INITIAL, {} }, |
|
[PPP_IPCP_EVENT_RCR_NEG] = { PPP_IPCP_STATE_INITIAL, {} }, |
|
[PPP_IPCP_EVENT_RCA] = { PPP_IPCP_STATE_INITIAL, {} }, |
|
[PPP_IPCP_EVENT_RCN] = { PPP_IPCP_STATE_INITIAL, {} }, |
|
[PPP_IPCP_EVENT_TO] = { PPP_IPCP_STATE_INITIAL, {} } |
|
}, |
|
[PPP_IPCP_STATE_REQ_SENT] = { |
|
[PPP_IPCP_EVENT_UP] = { PPP_IPCP_STATE_REQ_SENT, {} }, |
|
[PPP_IPCP_EVENT_DOWN] = { PPP_IPCP_STATE_INITIAL, {} }, |
|
[PPP_IPCP_EVENT_RCR_POS] = { PPP_IPCP_STATE_ACK_SENT, {ipcp_send_ack} }, |
|
[PPP_IPCP_EVENT_RCR_NEG] = { PPP_IPCP_STATE_REQ_SENT, {ipcp_send_nack} }, |
|
[PPP_IPCP_EVENT_RCA] = { PPP_IPCP_STATE_ACK_RCVD, {} }, |
|
[PPP_IPCP_EVENT_RCN] = { PPP_IPCP_STATE_REQ_SENT, {ipcp_send_req, ipcp_start_timer} }, |
|
[PPP_IPCP_EVENT_TO] = { PPP_IPCP_STATE_REQ_SENT, {ipcp_send_req, ipcp_start_timer} } |
|
}, |
|
[PPP_IPCP_STATE_ACK_RCVD] = { |
|
[PPP_IPCP_EVENT_UP] = { PPP_IPCP_STATE_ACK_RCVD, {} }, |
|
[PPP_IPCP_EVENT_DOWN] = { PPP_IPCP_STATE_INITIAL, {} }, |
|
[PPP_IPCP_EVENT_RCR_POS] = { PPP_IPCP_STATE_OPENED, {ipcp_send_ack, ipcp_bring_up} }, |
|
[PPP_IPCP_EVENT_RCR_NEG] = { PPP_IPCP_STATE_ACK_RCVD, {ipcp_send_nack} }, |
|
[PPP_IPCP_EVENT_RCA] = { PPP_IPCP_STATE_REQ_SENT, {ipcp_send_req, ipcp_start_timer} }, |
|
[PPP_IPCP_EVENT_RCN] = { PPP_IPCP_STATE_REQ_SENT, {ipcp_send_req, ipcp_start_timer} }, |
|
[PPP_IPCP_EVENT_TO] = { PPP_IPCP_STATE_ACK_RCVD, {ipcp_send_req, ipcp_start_timer} } |
|
}, |
|
[PPP_IPCP_STATE_ACK_SENT] = { |
|
[PPP_IPCP_EVENT_UP] = { PPP_IPCP_STATE_ACK_SENT, {} }, |
|
[PPP_IPCP_EVENT_DOWN] = { PPP_IPCP_STATE_INITIAL, {} }, |
|
[PPP_IPCP_EVENT_RCR_POS] = { PPP_IPCP_STATE_ACK_SENT, {ipcp_send_ack} }, |
|
[PPP_IPCP_EVENT_RCR_NEG] = { PPP_IPCP_STATE_REQ_SENT, {ipcp_send_nack} }, |
|
[PPP_IPCP_EVENT_RCA] = { PPP_IPCP_STATE_OPENED, {ipcp_bring_up} }, |
|
[PPP_IPCP_EVENT_RCN] = { PPP_IPCP_STATE_ACK_SENT, {ipcp_send_req, ipcp_start_timer} }, |
|
[PPP_IPCP_EVENT_TO] = { PPP_IPCP_STATE_ACK_SENT, {ipcp_send_req, ipcp_start_timer} } |
|
}, |
|
[PPP_IPCP_STATE_OPENED] = { |
|
[PPP_IPCP_EVENT_UP] = { PPP_IPCP_STATE_OPENED, {} }, |
|
[PPP_IPCP_EVENT_DOWN] = { PPP_IPCP_STATE_INITIAL, {ipcp_bring_down} }, |
|
[PPP_IPCP_EVENT_RCR_POS] = { PPP_IPCP_STATE_ACK_SENT, {ipcp_bring_down, ipcp_send_req, ipcp_send_ack} }, |
|
[PPP_IPCP_EVENT_RCR_NEG] = { PPP_IPCP_STATE_REQ_SENT, {ipcp_bring_down, ipcp_send_req, ipcp_send_nack} }, |
|
[PPP_IPCP_EVENT_RCA] = { PPP_IPCP_STATE_REQ_SENT, {ipcp_send_req} }, |
|
[PPP_IPCP_EVENT_RCN] = { PPP_IPCP_STATE_REQ_SENT, {ipcp_send_req} }, |
|
[PPP_IPCP_EVENT_TO] = { PPP_IPCP_STATE_OPENED, {} } |
|
} |
|
}; |
|
|
|
static void evaluate_ipcp_state(struct pico_device_ppp *ppp, enum ppp_ipcp_event event) |
|
{ |
|
const struct pico_ppp_fsm *fsm; |
|
int i; |
|
if (mock_ipcp_state) { |
|
mock_ipcp_state(ppp, event); |
|
return; |
|
} |
|
|
|
fsm = &ppp_ipcp_fsm[ppp->ipcp_state][event]; |
|
|
|
ppp->ipcp_state = (enum ppp_ipcp_state)fsm->next_state; |
|
for (i = 0; i < PPP_FSM_MAX_ACTIONS; i++) { |
|
if (fsm->event_handler[i]) |
|
fsm->event_handler[i](ppp); |
|
} |
|
} |
|
|
|
static int pico_ppp_poll(struct pico_device *dev, int loop_score) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *) dev; |
|
static uint32_t len = 0; |
|
int r; |
|
if (ppp->serial_recv) { |
|
do { |
|
r = ppp->serial_recv(&ppp->dev, &ppp_recv_buf[len], 1); |
|
if (r <= 0) |
|
break; |
|
|
|
if (ppp->modem_state == PPP_MODEM_STATE_CONNECTED) { |
|
static int control_escape = 0; |
|
|
|
if (ppp_recv_buf[len] == PPPF_FLAG_SEQ) { |
|
if (control_escape) { |
|
/* Illegal sequence, discard frame */ |
|
ppp_dbg("Illegal sequence, ppp_recv_buf[%d] = %d\n", len, ppp_recv_buf[len]); |
|
control_escape = 0; |
|
len = 0; |
|
} |
|
|
|
if (len > 1) { |
|
ppp_recv_data(ppp, ppp_recv_buf, len); |
|
loop_score--; |
|
len = 0; |
|
} |
|
} else if (control_escape) { |
|
ppp_recv_buf[len] ^= 0x20; |
|
control_escape = 0; |
|
len++; |
|
} else if (ppp_recv_buf[len] == PPPF_CTRL_ESC) { |
|
control_escape = 1; |
|
} else { |
|
len++; |
|
} |
|
} else { |
|
static int s3 = 0; |
|
|
|
if (ppp_recv_buf[len] == AT_S3) { |
|
s3 = 1; |
|
if (len > 0) { |
|
ppp_recv_buf[len] = '\0'; |
|
ppp_modem_recv(ppp, ppp_recv_buf, len); |
|
len = 0; |
|
} |
|
} else if (ppp_recv_buf[len] == AT_S4) { |
|
if (!s3) { |
|
len++; |
|
} |
|
|
|
s3 = 0; |
|
} else { |
|
s3 = 0; |
|
len++; |
|
} |
|
} |
|
} while ((r > 0) && (len < ARRAY_SIZE(ppp_recv_buf)) && (loop_score > 0)); |
|
} |
|
|
|
return loop_score; |
|
} |
|
|
|
/* Public interface: create/destroy. */ |
|
|
|
static int pico_ppp_link_state(struct pico_device *dev) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *)dev; |
|
if (ppp->ipcp_state == PPP_IPCP_STATE_OPENED) |
|
return 1; |
|
|
|
return 0; |
|
} |
|
|
|
void pico_ppp_destroy(struct pico_device *ppp) |
|
{ |
|
if (!ppp) |
|
return; |
|
|
|
/* Perform custom cleanup here before calling 'pico_device_destroy' |
|
* or register a custom cleanup function during initialization |
|
* by setting 'ppp->dev.destroy'. */ |
|
|
|
pico_device_destroy(ppp); |
|
} |
|
|
|
static void check_to_modem(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->timer_on & PPP_TIMER_ON_MODEM) { |
|
if (ppp->timer_val == 0) { |
|
ppp->timer_on = (uint8_t) (ppp->timer_on & (~PPP_TIMER_ON_MODEM)); |
|
evaluate_modem_state(ppp, PPP_MODEM_EVENT_TIMEOUT); |
|
} |
|
} |
|
} |
|
|
|
static void check_to_lcp(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->timer_on & (PPP_TIMER_ON_LCPREQ | PPP_TIMER_ON_LCPTERM)) { |
|
if (ppp->timer_val == 0) { |
|
if (ppp->timer_count == 0) |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_TO_NEG); |
|
else{ |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_TO_POS); |
|
ppp->timer_count--; |
|
} |
|
} |
|
} |
|
} |
|
|
|
static void check_to_auth(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->timer_on & PPP_TIMER_ON_AUTH) { |
|
if (ppp->timer_val == 0) { |
|
ppp->timer_on = (uint8_t) (ppp->timer_on & (~PPP_TIMER_ON_AUTH)); |
|
evaluate_auth_state(ppp, PPP_AUTH_EVENT_TO); |
|
} |
|
} |
|
} |
|
|
|
static void check_to_ipcp(struct pico_device_ppp *ppp) |
|
{ |
|
if (ppp->timer_on & PPP_TIMER_ON_IPCP) { |
|
if (ppp->timer_val == 0) { |
|
ppp->timer_on = (uint8_t) (ppp->timer_on & (~PPP_TIMER_ON_IPCP)); |
|
evaluate_ipcp_state(ppp, PPP_IPCP_EVENT_TO); |
|
} |
|
} |
|
} |
|
|
|
static void pico_ppp_tick(pico_time t, void *arg) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *) arg; |
|
(void)t; |
|
if (ppp->timer_val > 0) |
|
ppp->timer_val--; |
|
|
|
check_to_modem(ppp); |
|
check_to_lcp(ppp); |
|
check_to_auth(ppp); |
|
check_to_ipcp(ppp); |
|
|
|
if (ppp->autoreconnect && ppp->lcp_state == PPP_LCP_STATE_INITIAL) { |
|
ppp_dbg("(Re)connecting...\n"); |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_OPEN); |
|
} |
|
|
|
pico_timer_add(1000, pico_ppp_tick, arg); |
|
} |
|
|
|
struct pico_device *pico_ppp_create(void) |
|
{ |
|
struct pico_device_ppp *ppp = PICO_ZALLOC(sizeof(struct pico_device_ppp)); |
|
char devname[MAX_DEVICE_NAME]; |
|
|
|
if (!ppp) |
|
return NULL; |
|
|
|
snprintf(devname, MAX_DEVICE_NAME, "ppp%d", ppp_devnum++); |
|
|
|
if( 0 != pico_device_init((struct pico_device *)ppp, devname, NULL)) { |
|
return NULL; |
|
} |
|
|
|
ppp->dev.overhead = PPP_HDR_SIZE; |
|
ppp->dev.mtu = PICO_PPP_MTU; |
|
ppp->dev.send = pico_ppp_send; |
|
ppp->dev.poll = pico_ppp_poll; |
|
ppp->dev.link_state = pico_ppp_link_state; |
|
ppp->frame_id = (uint8_t)(pico_rand() % 0xFF); |
|
|
|
ppp->modem_state = PPP_MODEM_STATE_INITIAL; |
|
ppp->lcp_state = PPP_LCP_STATE_INITIAL; |
|
ppp->auth_state = PPP_AUTH_STATE_INITIAL; |
|
ppp->ipcp_state = PPP_IPCP_STATE_INITIAL; |
|
|
|
ppp->timer = pico_timer_add(1000, pico_ppp_tick, ppp); |
|
ppp->mru = PICO_PPP_MRU; |
|
|
|
LCPOPT_SET_LOCAL(ppp, LCPOPT_MRU); |
|
LCPOPT_SET_LOCAL(ppp, LCPOPT_AUTH); /* We support authentication, even if it's not part of the req */ |
|
LCPOPT_SET_LOCAL(ppp, LCPOPT_PROTO_COMP); |
|
LCPOPT_SET_LOCAL(ppp, LCPOPT_ADDRCTL_COMP); |
|
|
|
|
|
ppp_dbg("Device %s created.\n", ppp->dev.name); |
|
return (struct pico_device *)ppp; |
|
} |
|
|
|
int pico_ppp_connect(struct pico_device *dev) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *)dev; |
|
ppp->autoreconnect = 1; |
|
return 0; |
|
} |
|
|
|
int pico_ppp_disconnect(struct pico_device *dev) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *)dev; |
|
ppp->autoreconnect = 0; |
|
evaluate_lcp_state(ppp, PPP_LCP_EVENT_CLOSE); |
|
return 0; |
|
} |
|
|
|
int pico_ppp_set_serial_read(struct pico_device *dev, int (*sread)(struct pico_device *, void *, int)) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *)dev; |
|
|
|
if (!dev) |
|
return -1; |
|
|
|
ppp->serial_recv = sread; |
|
return 0; |
|
} |
|
|
|
int pico_ppp_set_serial_write(struct pico_device *dev, int (*swrite)(struct pico_device *, const void *, int)) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *)dev; |
|
|
|
if (!dev) |
|
return -1; |
|
|
|
ppp->serial_send = swrite; |
|
return 0; |
|
} |
|
|
|
int pico_ppp_set_serial_set_speed(struct pico_device *dev, int (*sspeed)(struct pico_device *, uint32_t)) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *)dev; |
|
|
|
if (!dev) |
|
return -1; |
|
|
|
ppp->serial_set_speed = sspeed; |
|
return 0; |
|
} |
|
|
|
int pico_ppp_set_apn(struct pico_device *dev, const char *apn) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *)dev; |
|
|
|
if (!dev) |
|
return -1; |
|
|
|
if (!apn) |
|
return -1; |
|
|
|
strncpy(ppp->apn, apn, sizeof(ppp->apn) - 1); |
|
return 0; |
|
} |
|
|
|
int pico_ppp_set_username(struct pico_device *dev, const char *username) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *)dev; |
|
|
|
if (!dev) |
|
return -1; |
|
|
|
if (!username) |
|
return -1; |
|
|
|
strncpy(ppp->username, username, sizeof(ppp->username) - 1); |
|
return 0; |
|
} |
|
|
|
int pico_ppp_set_password(struct pico_device *dev, const char *password) |
|
{ |
|
struct pico_device_ppp *ppp = (struct pico_device_ppp *)dev; |
|
|
|
if (!dev) |
|
return -1; |
|
|
|
if (!password) |
|
return -1; |
|
|
|
strncpy(ppp->password, password, sizeof(ppp->password) - 1); |
|
return 0; |
|
}
|
|
|