diff options
author | Kozlov Dmitry <xeb@mail.ru> | 2012-08-03 14:31:52 +0400 |
---|---|---|
committer | Kozlov Dmitry <xeb@mail.ru> | 2012-08-03 14:31:52 +0400 |
commit | 95586a13a28c93e60f8da6b728b0e7b57cdff5f1 (patch) | |
tree | af7e8c580068974b5b60d564b6fb45b7efef6ba0 | |
parent | ff80a772e3116afa4513397db2aa51962ff32994 (diff) | |
download | accel-ppp-xebd-95586a13a28c93e60f8da6b728b0e7b57cdff5f1.tar.gz accel-ppp-xebd-95586a13a28c93e60f8da6b728b0e7b57cdff5f1.zip |
ipoe: implemented dhcp relay
-rw-r--r-- | accel-pppd/accel-ppp.conf | 1 | ||||
-rw-r--r-- | accel-pppd/accel-ppp.conf.5 | 17 | ||||
-rw-r--r-- | accel-pppd/ctrl/ipoe/backup.c | 60 | ||||
-rw-r--r-- | accel-pppd/ctrl/ipoe/dhcpv4.c | 422 | ||||
-rw-r--r-- | accel-pppd/ctrl/ipoe/dhcpv4.h | 31 | ||||
-rw-r--r-- | accel-pppd/ctrl/ipoe/ipoe.c | 440 | ||||
-rw-r--r-- | accel-pppd/ctrl/ipoe/ipoe.h | 19 | ||||
-rw-r--r-- | accel-pppd/ctrl/ipoe/lua.c | 4 |
8 files changed, 754 insertions, 240 deletions
diff --git a/accel-pppd/accel-ppp.conf b/accel-pppd/accel-ppp.conf index 64c69e8..0199f05 100644 --- a/accel-pppd/accel-ppp.conf +++ b/accel-pppd/accel-ppp.conf @@ -45,6 +45,7 @@ shared=0 ifcfg=1 mode=L2 start=dhcpv4 +#relay=10.10.10.10 #attr-dhcp-client-ip=DHCP-Client-IP-Address #attr-dhcp-router-ip=DHCP-Router-IP-Address #attr-dhcp-mask=DHCP-Mask diff --git a/accel-pppd/accel-ppp.conf.5 b/accel-pppd/accel-ppp.conf.5 index 1dc9e85..82dd84a 100644 --- a/accel-pppd/accel-ppp.conf.5 +++ b/accel-pppd/accel-ppp.conf.5 @@ -229,8 +229,15 @@ Specifies default value for per-interface .B ifcfg parameter. .TP +.BI "relay=" ipv4_address +Specifies default value for per-interface +.B relay +parameter. +.TP .BI "interface=" [re:]name[,mode=L2|L3][,shared=0|1][,start=dhcpv4|up] .BI "" [,range=x.x.x.x/mask][,ifcfg=0|1] +.BI "" [,relay=x.x.x.x] +.BI "" [,giaddr=x.x.x.x] .br Specifies interface to listen dhcp or unclassified packets. You may specify multiple .B interface @@ -267,6 +274,16 @@ parameter specifies local range of ip address to give to dhcp clients. First IP The .B ifcfg parameter specifies whether accel-ppp should add router IP address and route to client to interface or it is explicitly configured. +.br +The +.B relay +parameter specifies DHCPv4 relay IP address to pass requests to. If specified +.B giaddr +is also needed. +.br +The +.B giaddr +parameter specifies relay agent IP address. .TP .BI "local-net=" x.x.x.x/mask Specifies networks from which packets will be treated as unclassified. You may specify multiple local-net options. diff --git a/accel-pppd/ctrl/ipoe/backup.c b/accel-pppd/ctrl/ipoe/backup.c index 0bc6448..9633bf7 100644 --- a/accel-pppd/ctrl/ipoe/backup.c +++ b/accel-pppd/ctrl/ipoe/backup.c @@ -15,8 +15,7 @@ #define IPOE_TAG_HWADDR 1 #define IPOE_TAG_CLIENT_ID 2 -#define IPOE_TAG_AGENT_CIRCUIT_ID 3 -#define IPOE_TAG_AGENT_REMOTE_ID 4 +#define IPOE_TAG_RELAY_AGENT 3 #define IPOE_TAG_XID 5 #define IPOE_TAG_GIADDR 6 #define IPOE_TAG_CALLING_SID 7 @@ -26,12 +25,16 @@ #define IPOE_TAG_YIADDR 11 #define IPOE_TAG_SIADDR 12 #define IPOE_TAG_MASK 13 +#define IPOE_TAG_RELAY_SERVER_ID 14 +#define IPOE_TAG_LEASE_TIME 15 #define IPOE_TAG_IFINDEX 100 -#define IPOE_FLAG_IFCFG 0x01 -#define IPOE_FLAG_DHCP_ADDR 0x02 -#define IPOE_FLAG_L4_REDIR 0x04 +#define IPOE_FLAG_IFCFG 0x01 +#define IPOE_FLAG_DHCP_ADDR 0x02 +#define IPOE_FLAG_RELAY_ADDR 0x04 +#define IPOE_FLAG_L4_REDIR 0x08 +#define IPOE_FLAG_L4_REDIR_SET 0x10 #define add_tag(id, data, size) if (!backup_add_tag(m, id, 0, data, size)) return -1; #define add_tag_i(id, data, size) if (!backup_add_tag(m, id, 1, data, size)) return -1; @@ -52,8 +55,14 @@ static int session_save(struct ap_session *ses, struct backup_mod *m) if (conn->dhcp_addr) flags |= IPOE_FLAG_DHCP_ADDR; + if (conn->relay_addr) + flags |= IPOE_FLAG_RELAY_ADDR; + if (conn->l4_redirect) flags |= IPOE_FLAG_L4_REDIR; + + if (conn->l4_redirect_set) + flags |= IPOE_FLAG_L4_REDIR_SET; add_tag(IPOE_TAG_HWADDR, conn->hwaddr, 6); add_tag(IPOE_TAG_CALLING_SID, ses->ctrl->calling_station_id, strlen(ses->ctrl->calling_station_id)); @@ -64,16 +73,16 @@ static int session_save(struct ap_session *ses, struct backup_mod *m) add_tag(IPOE_TAG_SIADDR, &conn->siaddr, 4); add_tag(IPOE_TAG_MASK, &conn->mask, 1); add_tag(IPOE_TAG_FLAGS, &flags, 4); + add_tag(IPOE_TAG_RELAY_SERVER_ID, &conn->relay_server_id, 4); + add_tag(IPOE_TAG_LEASE_TIME, &conn->lease_time, 4); + add_tag(IPOE_TAG_IFNAME, conn->serv->ifname, strlen(conn->serv->ifname) + 1); if (conn->client_id) add_tag(IPOE_TAG_CLIENT_ID, conn->client_id->data, conn->client_id->len); - if (conn->agent_circuit_id) - add_tag(IPOE_TAG_AGENT_CIRCUIT_ID, conn->agent_circuit_id->data, conn->agent_circuit_id->len); - if (conn->agent_circuit_id) - add_tag(IPOE_TAG_AGENT_REMOTE_ID, conn->agent_remote_id->data, conn->agent_remote_id->len); - - add_tag(IPOE_TAG_IFNAME, conn->serv->ifname, strlen(conn->serv->ifname) + 1); + if (conn->relay_agent) + add_tag(IPOE_TAG_RELAY_AGENT, conn->relay_agent->data, conn->relay_agent->len); + add_tag_i(IPOE_TAG_IFINDEX, &conn->ifindex, 4); return 0; @@ -87,9 +96,9 @@ static int session_restore(struct ap_session *ses, struct backup_mod *m) return 0; } -static void set_dhcpv4_opt(struct dhcp_opt **opt, struct backup_tag *t, uint8_t **ptr) +static void set_dhcpv4_opt(struct dhcpv4_option **opt, struct backup_tag *t, uint8_t **ptr) { - *opt = (struct dhcp_opt *)(*ptr); + *opt = (struct dhcpv4_option *)(*ptr); (*opt)->len = t->size; memcpy((*opt)->data, t->data, t->size); (*ptr) += sizeof(**opt) + t->size; @@ -112,9 +121,8 @@ static struct ap_session *ctrl_restore(struct backup_mod *m) list_for_each_entry(t, &m->tag_list, entry) { switch(t->id) { case IPOE_TAG_CLIENT_ID: - case IPOE_TAG_AGENT_CIRCUIT_ID: - case IPOE_TAG_AGENT_REMOTE_ID: - dlen += sizeof(struct dhcp_opt) + t->size; + case IPOE_TAG_RELAY_AGENT: + dlen += sizeof(struct dhcpv4_option) + t->size; break; case IPOE_TAG_IFNAME: ifname = t; @@ -162,11 +170,9 @@ static struct ap_session *ctrl_restore(struct backup_mod *m) case IPOE_TAG_CLIENT_ID: set_dhcpv4_opt(&ses->client_id, t, &ptr); break; - case IPOE_TAG_AGENT_CIRCUIT_ID: - set_dhcpv4_opt(&ses->agent_circuit_id, t, &ptr); - break; - case IPOE_TAG_AGENT_REMOTE_ID: - set_dhcpv4_opt(&ses->agent_remote_id, t, &ptr); + case IPOE_TAG_RELAY_AGENT: + set_dhcpv4_opt(&ses->relay_agent, t, &ptr); + dhcpv4_parse_opt82(ses->relay_agent, &ses->agent_circuit_id, &ses->agent_remote_id); break; case IPOE_TAG_IFINDEX: ses->ifindex = *(uint32_t *)t->data; @@ -183,6 +189,12 @@ static struct ap_session *ctrl_restore(struct backup_mod *m) case IPOE_TAG_FLAGS: flags = *(uint32_t *)t->data; break; + case IPOE_TAG_RELAY_SERVER_ID: + ses->relay_server_id = *(uint32_t *)t->data; + break; + case IPOE_TAG_LEASE_TIME: + ses->lease_time = *(uint32_t *)t->data; + break; } } @@ -194,8 +206,14 @@ static struct ap_session *ctrl_restore(struct backup_mod *m) ses->dhcp_addr = 1; } + if (flags & IPOE_FLAG_RELAY_ADDR) + ses->relay_addr = 1; + if (flags & IPOE_FLAG_L4_REDIR) ses->l4_redirect = 1; + + if (flags & IPOE_FLAG_L4_REDIR_SET && m->data->internal) + ses->l4_redirect = 1; ses->serv = serv; diff --git a/accel-pppd/ctrl/ipoe/dhcpv4.c b/accel-pppd/ctrl/ipoe/dhcpv4.c index 08da43a..86aba87 100644 --- a/accel-pppd/ctrl/ipoe/dhcpv4.c +++ b/accel-pppd/ctrl/ipoe/dhcpv4.c @@ -25,13 +25,14 @@ #include "dhcpv4.h" -#define DHCP_SERV_PORT 67 -#define DHCP_CLIENT_PORT 68 -#define DHCP_MAGIC "\x63\x82\x53\x63" - - #define BUF_SIZE 4096 +struct dhcpv4_relay_ctx +{ + struct list_head entry; + struct triton_context_t *ctx; + triton_event_func recv; +}; static int conf_verbose; static in_addr_t conf_dns1; @@ -40,6 +41,8 @@ static in_addr_t conf_dns2; static mempool_t pack_pool; static mempool_t opt_pool; +static LIST_HEAD(relay_list); + static int dhcpv4_read(struct triton_md_handler_t *h); static struct dhcpv4_iprange *parse_range(const char *str) @@ -228,11 +231,11 @@ void dhcpv4_free(struct dhcpv4_serv *serv) _free(serv); } -void dhcpv4_print_packet(struct dhcpv4_packet *pack, void (*print)(const char *fmt, ...)) +void dhcpv4_print_packet(struct dhcpv4_packet *pack, int relay, void (*print)(const char *fmt, ...)) { const char *msg_name[] = {"Discover", "Offer", "Request", "Decline", "Ack", "Nak", "Release", "Inform"}; - print("[DHCPv4 %s xid=%x ", msg_name[pack->msg_type - 1], pack->hdr->xid); + print("[DHCPv4 %s%s xid=%x ", relay ? "relay " : "", msg_name[pack->msg_type - 1], pack->hdr->xid); if (pack->hdr->ciaddr) print("ciaddr=%i.%i.%i.%i ", @@ -276,41 +279,6 @@ void dhcpv4_print_packet(struct dhcpv4_packet *pack, void (*print)(const char *f print("]\n"); } -static int parse_opt82(struct dhcpv4_packet *pack, struct dhcpv4_option *opt) -{ - uint8_t *ptr = opt->data; - uint8_t *endptr = ptr + opt->len; - int type, len; - struct dhcpv4_option *opt1; - - while (ptr < endptr) { - type = *ptr++; - len = *ptr++; - if (ptr + len > endptr) - return -1; - if (type == 1 || type == 2) { - opt1 = mempool_alloc(opt_pool); - if (!opt1) { - log_emerg("out of memory\n"); - return -1; - } - - opt1->type = type; - opt1->len = len; - opt1->data = ptr; - - if (type == 1) - pack->agent_circuit_id = opt1; - else - pack->agent_remote_id = opt1; - } - - ptr += len; - } - - return 0; -} - static int dhcpv4_parse_packet(struct dhcpv4_packet *pack, int len) { struct dhcpv4_option *opt; @@ -321,9 +289,6 @@ static int dhcpv4_parse_packet(struct dhcpv4_packet *pack, int len) log_warn("dhcpv4: short packet received\n"); return -1; } - - if (pack->hdr->op != DHCP_OP_REQUEST) - return -1; if (pack->hdr->htype != 1) return -1; @@ -342,8 +307,10 @@ static int dhcpv4_parse_packet(struct dhcpv4_packet *pack, int len) continue; } - if (*ptr == 0xff) + if (*ptr == 0xff) { + ptr++; break; + } opt = mempool_alloc(opt_pool); if (!opt) { @@ -364,7 +331,9 @@ static int dhcpv4_parse_packet(struct dhcpv4_packet *pack, int len) if (opt->type == 53) pack->msg_type = opt->data[0]; else if (opt->type == 82) - parse_opt82(pack, opt); + pack->relay_agent = opt; + else if (opt->type == 62) + pack->client_id = opt; else if (opt->type == 50) pack->request_ip = *(uint32_t *)opt->data; else if (opt->type == 54) @@ -377,6 +346,8 @@ static int dhcpv4_parse_packet(struct dhcpv4_packet *pack, int len) if (dhcpv4_check_options(pack)) return -1; + pack->ptr = ptr; + /*if (conf_verbose) { log_info2("recv "); print_packet(pack, log_info2); @@ -398,12 +369,71 @@ static struct dhcpv4_packet *dhcpv4_packet_alloc() pack->hdr = (struct dhcpv4_hdr *)pack->data; pack->ptr = (uint8_t *)(pack->hdr + 1); + pack->refs = 1; memcpy(pack->hdr->magic, DHCP_MAGIC, 4); return pack; } +void dhcpv4_packet_ref(struct dhcpv4_packet *pack) +{ + __sync_add_and_fetch(&pack->refs, 1); +} + +struct dhcpv4_option *dhcpv4_packet_find_opt(struct dhcpv4_packet *pack, int type) +{ + struct dhcpv4_option *opt; + + list_for_each_entry(opt, &pack->options, entry) { + if (opt->type == type) + return opt; + } + + return NULL; +} + +void dhcpv4_packet_free(struct dhcpv4_packet *pack) +{ + struct dhcpv4_option *opt; + + if (__sync_sub_and_fetch(&pack->refs, 1)) + return; + + while (!list_empty(&pack->options)) { + opt = list_entry(pack->options.next, typeof(*opt), entry); + list_del(&opt->entry); + mempool_free(opt); + } + + mempool_free(pack); +} + +int dhcpv4_parse_opt82(struct dhcpv4_option *opt, uint8_t **agent_circuit_id, uint8_t **agent_remote_id) +{ + uint8_t *ptr = opt->data; + uint8_t *endptr = ptr + opt->len; + int type, len; + + while (ptr < endptr) { + type = *ptr++; + len = *ptr++; + + if (ptr + len > endptr) + return -1; + + if (type == 1) + *agent_circuit_id = ptr - 1; + else if (type == 2) + *agent_remote_id = ptr - 1; + + ptr += len; + } + + return 0; +} + + static int dhcpv4_read(struct triton_md_handler_t *h) { struct dhcpv4_packet *pack; @@ -433,12 +463,62 @@ static int dhcpv4_read(struct triton_md_handler_t *h) dhcpv4_packet_free(pack); continue; } + + if (pack->hdr->op != DHCP_OP_REQUEST) { + dhcpv4_packet_free(pack); + continue; + } if (serv->recv) serv->recv(serv, pack); + + dhcpv4_packet_free(pack); + } +} + +static int dhcpv4_relay_read(struct triton_md_handler_t *h) +{ + struct dhcpv4_packet *pack; + struct dhcpv4_relay *r = container_of(h, typeof(*r), hnd); + int n; + struct dhcpv4_relay_ctx *c; + + while (1) { + pack = dhcpv4_packet_alloc(); + if (!pack) { + log_emerg("out of memory\n"); + return 1; + } + + n = read(h->fd, pack->data, BUF_SIZE); + if (n == -1) { + mempool_free(pack); + if (errno == EAGAIN) + return 0; + log_error("dhcpv4: recv: %s\n", strerror(errno)); + continue; + } + + if (dhcpv4_parse_packet(pack, n)) { + dhcpv4_packet_free(pack); + continue; + } + + if (pack->hdr->op != DHCP_OP_REPLY) { + dhcpv4_packet_free(pack); + continue; + } + + list_for_each_entry(c, &r->ctx_list, entry) { + dhcpv4_packet_ref(pack); + triton_context_call(c->ctx, c->recv, pack); + } + + dhcpv4_packet_free(pack); } } + uint16_t ip_csum(uint16_t *buf, int len) { uint32_t sum=0; @@ -528,25 +608,6 @@ static int dhcpv4_send(struct dhcpv4_serv *serv, struct dhcpv4_packet *pack, in_ return dhcpv4_send_raw(serv, pack, saddr, daddr); } -void dhcpv4_packet_free(struct dhcpv4_packet *pack) -{ - struct dhcpv4_option *opt; - - while (!list_empty(&pack->options)) { - opt = list_entry(pack->options.next, typeof(*opt), entry); - list_del(&opt->entry); - mempool_free(opt); - } - - if (pack->agent_circuit_id) - mempool_free(pack->agent_circuit_id); - - if (pack->agent_remote_id) - mempool_free(pack->agent_remote_id); - - mempool_free(pack); -} - int dhcpv4_packet_add_opt(struct dhcpv4_packet *pack, int type, const void *data, int len) { struct dhcpv4_option *opt = mempool_alloc(opt_pool); @@ -571,7 +632,7 @@ int dhcpv4_packet_add_opt(struct dhcpv4_packet *pack, int type, const void *data return 0; } -int dhcpv4_send_reply(int msg_type, struct dhcpv4_serv *serv, struct dhcpv4_packet *req, uint32_t yiaddr, uint32_t siaddr, uint32_t mask, int lease_time) +int dhcpv4_send_reply(int msg_type, struct dhcpv4_serv *serv, struct dhcpv4_packet *req, uint32_t yiaddr, uint32_t siaddr, uint32_t mask, int lease_time, struct dhcpv4_packet *relay) { struct dhcpv4_packet *pack; int val, r; @@ -579,6 +640,8 @@ int dhcpv4_send_reply(int msg_type, struct dhcpv4_serv *serv, struct dhcpv4_pack in_addr_t dns1; in_addr_t dns2; } dns; + int dns_avail = 0; + struct dhcpv4_option *opt; pack = dhcpv4_packet_alloc(); if (!pack) { @@ -589,7 +652,7 @@ int dhcpv4_send_reply(int msg_type, struct dhcpv4_serv *serv, struct dhcpv4_pack memcpy(pack->hdr, req->hdr, sizeof(*req->hdr)); pack->hdr->op = DHCP_OP_REPLY; - pack->hdr->ciaddr = 0; + //pack->hdr->ciaddr = 0; pack->hdr->yiaddr = yiaddr; if (msg_type == DHCPOFFER) pack->hdr->siaddr = siaddr; @@ -612,15 +675,28 @@ int dhcpv4_send_reply(int msg_type, struct dhcpv4_serv *serv, struct dhcpv4_pack val = htonl(~((1 << (32 - mask)) - 1)); if (dhcpv4_packet_add_opt(pack, 1, &val, 4)) goto out_err; + + if (relay) { + list_for_each_entry(opt, &relay->options, entry) { + if (opt->type == 53 || opt->type == 54 || opt->type == 51 || opt->type == 3 || opt->type == 1) + continue; + if (opt->type == 6) + dns_avail = 1; + if (dhcpv4_packet_add_opt(pack, opt->type, opt->data, opt->len)) + goto out_err; + } + } - if (conf_dns1 && conf_dns2) { - dns.dns1 = conf_dns1; - dns.dns2 = conf_dns2; - if (dhcpv4_packet_add_opt(pack, 6, &dns, 8)) - goto out_err; - } else if (conf_dns1) { - if (dhcpv4_packet_add_opt(pack, 6, &conf_dns1, 4)) - goto out_err; + if (!dns_avail) { + if (conf_dns1 && conf_dns2) { + dns.dns1 = conf_dns1; + dns.dns2 = conf_dns2; + if (dhcpv4_packet_add_opt(pack, 6, &dns, 8)) + goto out_err; + } else if (conf_dns1) { + if (dhcpv4_packet_add_opt(pack, 6, &conf_dns1, 4)) + goto out_err; + } } *pack->ptr++ = 255; @@ -628,7 +704,7 @@ int dhcpv4_send_reply(int msg_type, struct dhcpv4_serv *serv, struct dhcpv4_pack if (conf_verbose) { pack->msg_type = msg_type; log_ppp_info2("send "); - dhcpv4_print_packet(pack, log_ppp_info2); + dhcpv4_print_packet(pack, 0, log_ppp_info2); } r = dhcpv4_send(serv, pack, siaddr, yiaddr); @@ -669,7 +745,7 @@ int dhcpv4_send_nak(struct dhcpv4_serv *serv, struct dhcpv4_packet *req) if (conf_verbose) { pack->msg_type = DHCPNAK; log_info2("send "); - dhcpv4_print_packet(pack, log_info2); + dhcpv4_print_packet(pack, 0, log_info2); } r = dhcpv4_send(serv, pack, 0, 0xffffffff); @@ -685,6 +761,196 @@ out_err: return 0; } +struct dhcpv4_relay *dhcpv4_relay_create(const char *_addr, const char *_giaddr, struct triton_context_t *ctx, triton_event_func recv) +{ + struct dhcpv4_relay *r; + in_addr_t addr = inet_addr(_addr); + in_addr_t giaddr = inet_addr(_giaddr); + struct sockaddr_in raddr; + struct sockaddr_in laddr; + int sock = -1; + int f = 1; + struct dhcpv4_relay_ctx *c; + + list_for_each_entry(r, &relay_list, entry) { + if (r->addr == addr && r->giaddr == giaddr) + goto found; + } + + r = _malloc(sizeof(*r)); + memset(r, 0, sizeof(*r)); + INIT_LIST_HEAD(&r->ctx_list); + r->addr = addr; + r->giaddr = giaddr; + + memset(&raddr, 0, sizeof(raddr)); + raddr.sin_family = AF_INET; + raddr.sin_addr.s_addr = addr; + raddr.sin_port = htons(DHCP_SERV_PORT); + + memset(&laddr, 0, sizeof(laddr)); + laddr.sin_family = AF_INET; + laddr.sin_addr.s_addr = giaddr; + laddr.sin_port = htons(DHCP_SERV_PORT); + + sock = socket(PF_INET, SOCK_DGRAM, IPPROTO_UDP); + if (!sock) { + log_error("socket: %s\n", strerror(errno)); + goto out_err; + } + + if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &f, sizeof(f))) + log_error("dhcpv4: setsockopt(SO_REUSEADDR): %s\n", strerror(errno)); + + if (bind(sock, &laddr, sizeof(laddr))) { + log_error("dhcpv4: relay: %s: bind: %s\n", _addr, strerror(errno)); + goto out_err; + } + + if (connect(sock, &raddr, sizeof(raddr))) { + log_error("dhcpv4: relay: %s: connect: %s\n", _addr, strerror(errno)); + goto out_err; + } + + fcntl(sock, F_SETFL, O_NONBLOCK); + fcntl(sock, F_SETFD, fcntl(sock, F_GETFD) | FD_CLOEXEC); + + r->hnd.fd = sock; + r->hnd.read = dhcpv4_relay_read; + + triton_context_register(&r->ctx, NULL); + triton_md_register_handler(ctx, &r->hnd); + triton_md_enable_handler(&r->hnd, MD_MODE_READ); + triton_context_wakeup(&r->ctx); + + list_add_tail(&r->entry, &relay_list); + +found: + c = _malloc(sizeof(*c)); + c->ctx = ctx; + c->recv = recv; + list_add_tail(&c->entry, &r->ctx_list); + + return r; + +out_err: + if (sock != -1) + close(sock); + _free(r); + return NULL; +} + +void dhcpv4_relay_free(struct dhcpv4_relay *r, struct triton_context_t *ctx) +{ + struct dhcpv4_relay_ctx *c; + + list_for_each_entry(c, &r->ctx_list, entry) { + if (c->ctx == ctx) { + list_del(&c->entry); + _free(c); + break; + } + } + + if (list_empty(&r->ctx_list)) { + list_del(&r->entry); + + triton_md_unregister_handler(&r->hnd); + close(r->hnd.fd); + triton_context_unregister(&r->ctx); + _free(r); + } +} + +int dhcpv4_relay_send(struct dhcpv4_relay *relay, struct dhcpv4_packet *request, uint32_t server_id) +{ + int n; + int len = request->ptr - request->data; + uint32_t giaddr = request->hdr->giaddr; + struct dhcpv4_option *opt = NULL; + uint32_t _server_id; + + request->hdr->giaddr = relay->giaddr; + + if (server_id) { + opt = dhcpv4_packet_find_opt(request, 54); + if (opt) { + _server_id = *(uint32_t *)opt->data; + *(uint32_t *)opt->data = server_id; + } + } + + if (conf_verbose) { + log_ppp_info2("send "); + dhcpv4_print_packet(request, 1, log_ppp_info2); + } + + n = write(relay->hnd.fd, request->data, len); + + request->hdr->giaddr = giaddr; + + if (opt) + *(uint32_t *)opt->data = _server_id; + + if (n != len) + return -1; + + return 0; +} + +int dhcpv4_relay_send_release(struct dhcpv4_relay *relay, uint8_t *chaddr, uint32_t xid, uint32_t ciaddr, + struct dhcpv4_option *client_id, struct dhcpv4_option *relay_agent) +{ + struct dhcpv4_packet *pack; + int n, len; + + pack = dhcpv4_packet_alloc(); + if (!pack) { + log_emerg("out of memory\n"); + return -1; + } + + memset(pack->hdr, 0, sizeof(*pack->hdr)); + + pack->msg_type = DHCPRELEASE; + pack->hdr->op = DHCP_OP_REQUEST; + pack->hdr->htype = 1; + pack->hdr->hlen = 6; + pack->hdr->ciaddr = ciaddr; + pack->hdr->giaddr = relay->giaddr; + pack->hdr->xid = xid; + memcpy(pack->hdr->magic, DHCP_MAGIC, 4); + memcpy(pack->hdr->chaddr, chaddr, 6); + + if (dhcpv4_packet_add_opt(pack, 53, &pack->msg_type, 1)) + goto out_err; + + if (client_id && dhcpv4_packet_add_opt(pack, 61, client_id->data, client_id->len)) + goto out_err; + + if (relay_agent && dhcpv4_packet_add_opt(pack, 82, relay_agent->data, relay_agent->len)) + goto out_err; + + *pack->ptr = 255; pack->ptr++; + + len = pack->ptr - pack->data; + + if (conf_verbose) { + log_ppp_info2("send "); + dhcpv4_print_packet(pack, 1, log_ppp_info2); + } + + n = write(relay->hnd.fd, pack->data, len); + + dhcpv4_packet_free(pack); + + return n == len ? 0 : -1; + +out_err: + dhcpv4_packet_free(pack); + return -1; +} + int dhcpv4_get_ip(struct dhcpv4_serv *serv, uint32_t *yiaddr, uint32_t *siaddr, int *mask) { int i, k; diff --git a/accel-pppd/ctrl/ipoe/dhcpv4.h b/accel-pppd/ctrl/ipoe/dhcpv4.h index 3ebb74c..67f7e61 100644 --- a/accel-pppd/ctrl/ipoe/dhcpv4.h +++ b/accel-pppd/ctrl/ipoe/dhcpv4.h @@ -9,6 +9,10 @@ #define __packed __attribute__((packed)) +#define DHCP_SERV_PORT 67 +#define DHCP_CLIENT_PORT 68 +#define DHCP_MAGIC "\x63\x82\x53\x63" + #define DHCP_OP_REQUEST 1 #define DHCP_OP_REPLY 2 @@ -53,11 +57,11 @@ struct dhcpv4_packet struct dhcpv4_hdr *hdr; struct list_head options; struct dhcpv4_option *client_id; - struct dhcpv4_option *agent_circuit_id; - struct dhcpv4_option *agent_remote_id; + struct dhcpv4_option *relay_agent; uint32_t request_ip; uint32_t server_id; int msg_type; + int volatile refs; uint8_t *ptr; uint8_t data[0]; }; @@ -84,21 +88,40 @@ struct dhcpv4_serv struct dhcpv4_iprange *range; }; +struct dhcpv4_relay +{ + struct list_head entry; + struct triton_context_t ctx; + struct triton_md_handler_t hnd; + struct list_head ctx_list; + in_addr_t addr; + in_addr_t giaddr; +}; + struct ap_session; struct dhcpv4_serv *dhcpv4_create(struct triton_context_t *ctx, const char *ifname, const char *opt); void dhcpv4_free(struct dhcpv4_serv *); +struct dhcpv4_relay *dhcpv4_relay_create(const char *addr, const char *giaddr, struct triton_context_t *ctx, triton_event_func recv); +void dhcpv4_relay_free(struct dhcpv4_relay *, struct triton_context_t *); +int dhcpv4_relay_send(struct dhcpv4_relay *relay, struct dhcpv4_packet *request, uint32_t server_id); +int dhcpv4_relay_send_release(struct dhcpv4_relay *relay, uint8_t *chaddr, uint32_t xid, uint32_t ciaddr, + struct dhcpv4_option *client_id, struct dhcpv4_option *relay_agent); -int dhcpv4_send_reply(int msg_type, struct dhcpv4_serv *serv, struct dhcpv4_packet *req, uint32_t yiaddr, uint32_t siaddr, uint32_t mask, int lease_time); +int dhcpv4_send_reply(int msg_type, struct dhcpv4_serv *serv, struct dhcpv4_packet *req, uint32_t yiaddr, uint32_t siaddr, uint32_t mask, int lease_time, struct dhcpv4_packet *relay_reply); int dhcpv4_send_nak(struct dhcpv4_serv *serv, struct dhcpv4_packet *req); +void dhcpv4_packet_ref(struct dhcpv4_packet *pack); +struct dhcpv4_option *dhcpv4_packet_find_opt(struct dhcpv4_packet *pack, int type); void dhcpv4_packet_free(struct dhcpv4_packet *pack); int dhcpv4_check_options(struct dhcpv4_packet *); void dhcpv4_print_options(struct dhcpv4_packet *, void (*)(const char *, ...)); -void dhcpv4_print_packet(struct dhcpv4_packet *pack, void (*print)(const char *fmt, ...)); +void dhcpv4_print_packet(struct dhcpv4_packet *pack, int relay, void (*print)(const char *fmt, ...)); + +int dhcpv4_parse_opt82(struct dhcpv4_option *opt, uint8_t **agent_circuit_id, uint8_t **agent_remote_id); int dhcpv4_get_ip(struct dhcpv4_serv *serv, uint32_t *yiaddr, uint32_t *siaddr, int *mask); void dhcpv4_put_ip(struct dhcpv4_serv *serv, uint32_t ip); diff --git a/accel-pppd/ctrl/ipoe/ipoe.c b/accel-pppd/ctrl/ipoe/ipoe.c index 711dca3..14e8d37 100644 --- a/accel-pppd/ctrl/ipoe/ipoe.c +++ b/accel-pppd/ctrl/ipoe/ipoe.c @@ -59,6 +59,7 @@ static int conf_attr_dhcp_mask; static int conf_attr_l4_redirect; #endif static int conf_l4_redirect_table; +static const char *conf_relay; #ifdef USE_LUA static const char *conf_lua_username_func; @@ -109,46 +110,54 @@ static void ipoe_serv_close(struct triton_context_t *ctx); static struct ipoe_session *ipoe_session_lookup(struct ipoe_serv *serv, struct dhcpv4_packet *pack) { struct ipoe_session *ses; - struct ipoe_session *ses1 = NULL; + + uint8_t *agent_circuit_id = NULL; + uint8_t *agent_remote_id = NULL; - list_for_each_entry(ses, &serv->sessions, entry) { - if (pack->hdr->giaddr != ses->giaddr) - continue; + if (pack->relay_agent && dhcpv4_parse_opt82(pack->relay_agent, &agent_circuit_id, &agent_remote_id)) { + agent_circuit_id = NULL; + agent_remote_id = NULL; + } - if (pack->agent_circuit_id && !ses->agent_circuit_id) + list_for_each_entry(ses, &serv->sessions, entry) { + if (agent_circuit_id && !ses->agent_circuit_id) continue; - if (pack->agent_remote_id && !ses->agent_remote_id) + if (agent_remote_id && !ses->agent_remote_id) continue; - if (!pack->agent_circuit_id && ses->agent_circuit_id) + if (!agent_circuit_id && ses->agent_circuit_id) continue; - if (!pack->agent_remote_id && ses->agent_remote_id) + if (!agent_remote_id && ses->agent_remote_id) continue; - if (pack->agent_circuit_id) { - if (pack->agent_circuit_id->len != ses->agent_circuit_id->len) + if (agent_circuit_id) { + if (*agent_circuit_id != *ses->agent_circuit_id) continue; - if (memcmp(pack->agent_circuit_id->data, ses->agent_circuit_id->data, pack->agent_circuit_id->len)) + if (memcmp(agent_circuit_id + 1, ses->agent_circuit_id + 1, *agent_circuit_id)) continue; } - if (pack->agent_remote_id) { - if (pack->agent_remote_id->len != ses->agent_remote_id->len) + if (agent_remote_id) { + if (*agent_remote_id != *ses->agent_remote_id) continue; - if (memcmp(pack->agent_remote_id->data, ses->agent_remote_id->data, pack->agent_remote_id->len)) + if (memcmp(agent_remote_id + 1, ses->agent_remote_id + 1, *agent_remote_id)) continue; - + return ses; } + + if (memcmp(pack->hdr->chaddr, ses->hwaddr, 6)) + continue; + + return ses; - if (pack->client_id && !ses->client_id) + /*if (pack->client_id && !ses->client_id) continue; if (!pack->client_id && ses->client_id) continue; - if (pack->client_id) { if (pack->client_id->len != ses->client_id->len) @@ -157,18 +166,15 @@ static struct ipoe_session *ipoe_session_lookup(struct ipoe_serv *serv, struct d continue; } - if (memcmp(pack->hdr->chaddr, ses->hwaddr, 6)) - continue; - ses1 = ses; if (pack->hdr->xid != ses->xid) continue; - return ses; + return ses;*/ } - return ses1; + return NULL; } static void ipoe_session_timeout(struct triton_timer_t *t) @@ -177,7 +183,7 @@ static void ipoe_session_timeout(struct triton_timer_t *t) triton_timer_del(t); - log_ppp_info2("session timed out\n"); + log_ppp_info2("ipoe: session timed out\n"); ap_session_terminate(&ses->ses, TERM_LOST_CARRIER, 0); } @@ -204,10 +210,13 @@ static void ipoe_change_l4_redirect(struct ipoe_session *ses, int del) else addr = ses->yiaddr; - if (del) + if (del) { iprule_del(addr, conf_l4_redirect_table); - else + ses->l4_redirect_set = 0; + } else { iprule_add(addr, conf_l4_redirect_table); + ses->l4_redirect_set = 1; + } } static void ipoe_change_addr(struct ipoe_session *ses, in_addr_t newaddr) @@ -215,6 +224,7 @@ static void ipoe_change_addr(struct ipoe_session *ses, in_addr_t newaddr) } +static void __ipoe_session_start(struct ipoe_session *ses); static void ipoe_session_start(struct ipoe_session *ses) { int r; @@ -225,6 +235,7 @@ static void ipoe_session_start(struct ipoe_session *ses) if (ses->serv->opt_shared == 0 && (!ses->ses.ipv4 || ses->ses.ipv4->peer_addr == ses->yiaddr)) { strncpy(ses->ses.ifname, ses->serv->ifname, AP_IFNAME_LEN); ses->ses.ifindex = ses->serv->ifindex; + ses->ctrl.dont_ifcfg = 1; } else if (ses->ifindex == -1) { pthread_mutex_lock(&uc_lock); if (!list_empty(&uc_list)) { @@ -289,6 +300,18 @@ static void ipoe_session_start(struct ipoe_session *ses) return; } + if (ses->dhcpv4_request && ses->serv->dhcpv4_relay) { + dhcpv4_relay_send(ses->serv->dhcpv4_relay, ses->dhcpv4_request, ses->relay_server_id); + + ses->timer.expire = ipoe_session_timeout; + ses->timer.expire_tv.tv_sec = conf_offer_timeout; + triton_timer_add(&ses->ctx, &ses->timer, 0); + } else + __ipoe_session_start(ses); +} + +static void __ipoe_session_start(struct ipoe_session *ses) +{ if (!ses->yiaddr) { dhcpv4_get_ip(ses->serv->dhcpv4, &ses->yiaddr, &ses->siaddr, &ses->mask); if (ses->yiaddr) @@ -320,9 +343,9 @@ static void ipoe_session_start(struct ipoe_session *ses) if (!ses->mask) ses->mask = 24; - + if (ses->dhcpv4_request) { - dhcpv4_send_reply(DHCPOFFER, ses->serv->dhcpv4, ses->dhcpv4_request, ses->yiaddr, ses->siaddr, ses->mask, conf_lease_time); + dhcpv4_send_reply(DHCPOFFER, ses->serv->dhcpv4, ses->dhcpv4_request, ses->yiaddr, ses->siaddr, ses->mask, ses->lease_time, ses->dhcpv4_relay_reply); dhcpv4_packet_free(ses->dhcpv4_request); ses->dhcpv4_request = NULL; @@ -338,38 +361,66 @@ static void ipoe_session_start(struct ipoe_session *ses) } } -static void ipoe_ifcfg_add(struct ipoe_session *ses) +static void ipoe_serv_add_addr(struct ipoe_serv *serv, in_addr_t addr) { struct ifaddr *a; - struct ipoe_serv *serv = ses->serv; - int f = 0; pthread_mutex_lock(&serv->lock); - if (ses->serv->opt_shared) { - list_for_each_entry(a, &serv->addr_list, entry) { - if (a->addr == ses->siaddr) { - f = 1; - break; + list_for_each_entry(a, &serv->addr_list, entry) { + if (a->addr == addr) { + a->refs++; + pthread_mutex_unlock(&serv->lock); + + return; + } + } + + a = _malloc(sizeof(*a)); + a->addr = addr; + a->refs = 1; + list_add_tail(&a->entry, &serv->addr_list); + + if (ipaddr_add(serv->ifindex, a->addr, 32)) + log_warn("ipoe: failed to add addess to interface '%s'\n", serv->ifname); + + pthread_mutex_unlock(&serv->lock); +} + +static void ipoe_serv_del_addr(struct ipoe_serv *serv, in_addr_t addr) +{ + struct ifaddr *a; + + pthread_mutex_lock(&serv->lock); + + list_for_each_entry(a, &serv->addr_list, entry) { + if (a->addr == addr) { + if (--a->refs == 0) { + if (ipaddr_del(serv->ifindex, a->addr)) + log_warn("ipoe: failed to delete addess from interface '%s'\n", serv->ifname); + list_del(&a->entry); + _free(a); } + break; } - if (!f) { - a = _malloc(sizeof(*a)); - a->addr = ses->siaddr; - a->refs = 1; - list_add_tail(&a->entry, &serv->addr_list); - - if (ipaddr_add(serv->ifindex, a->addr, 32)) - log_ppp_warn("ipoe: failed to add addess to interface '%s'\n", serv->ifname); - } else - a->refs++; - } else { + } + + pthread_mutex_unlock(&serv->lock); +} + +static void ipoe_ifcfg_add(struct ipoe_session *ses) +{ + struct ipoe_serv *serv = ses->serv; + + if (ses->serv->opt_shared || ses->serv->dhcpv4_relay) + ipoe_serv_add_addr(ses->serv, ses->siaddr); + else { + pthread_mutex_lock(&serv->lock); if (ipaddr_add(serv->ifindex, ses->siaddr, 32)) log_ppp_warn("ipoe: failed to add addess to interface '%s'\n", serv->ifname); + pthread_mutex_unlock(&serv->lock); } - pthread_mutex_unlock(&serv->lock); - if (iproute_add(serv->ifindex, ses->siaddr, ses->yiaddr)) log_ppp_warn("ipoe: failed to add route to interface '%s'\n", serv->ifname); @@ -378,34 +429,22 @@ static void ipoe_ifcfg_add(struct ipoe_session *ses) static void ipoe_ifcfg_del(struct ipoe_session *ses) { - struct ifaddr *a; struct ipoe_serv *serv = ses->serv; if (iproute_del(serv->ifindex, ses->yiaddr)) log_ppp_warn("ipoe: failed to delete route from interface '%s'\n", serv->ifname); - pthread_mutex_lock(&serv->lock); - - if (ses->serv->opt_shared) { - list_for_each_entry(a, &serv->addr_list, entry) { - if (a->addr == ses->siaddr) - break; - } - if (--a->refs == 0) { - if (ipaddr_del(serv->ifindex, a->addr)) - log_ppp_warn("ipoe: failed to delete addess from interface '%s'\n", serv->ifname); - list_del(&a->entry); - _free(a); - } + if (ses->serv->opt_shared || ses->serv->dhcpv4_relay) { + ipoe_serv_del_addr(ses->serv, ses->siaddr); } else { + pthread_mutex_lock(&serv->lock); if (ipaddr_del(serv->ifindex, ses->siaddr)) log_ppp_warn("ipoe: failed to add addess to interface '%s'\n", serv->ifname); + pthread_mutex_unlock(&serv->lock); } - - pthread_mutex_unlock(&serv->lock); } -static void ipoe_session_activate(struct ipoe_session *ses) +static void __ipoe_session_activate(struct ipoe_session *ses) { uint32_t addr; @@ -420,8 +459,7 @@ static void ipoe_session_activate(struct ipoe_session *ses) ap_session_terminate(&ses->ses, TERM_NAS_ERROR, 0); return; } - } else - ses->ctrl.dont_ifcfg = 1; + } if (ses->serv->opt_ifcfg) ipoe_ifcfg_add(ses); @@ -433,7 +471,7 @@ static void ipoe_session_activate(struct ipoe_session *ses) if (ses->dhcpv4_request) { if (ses->ses.state == AP_STATE_ACTIVE) - dhcpv4_send_reply(DHCPACK, ses->serv->dhcpv4, ses->dhcpv4_request, ses->yiaddr, ses->siaddr, ses->mask, conf_lease_time); + dhcpv4_send_reply(DHCPACK, ses->serv->dhcpv4, ses->dhcpv4_request, ses->yiaddr, ses->siaddr, ses->mask, ses->lease_time, ses->dhcpv4_relay_reply); else dhcpv4_send_nak(ses->serv->dhcpv4, ses->dhcpv4_request); @@ -442,16 +480,36 @@ static void ipoe_session_activate(struct ipoe_session *ses) } } -static void ipoe_session_keepalive(struct ipoe_session *ses) +static void ipoe_session_activate(struct ipoe_session *ses) +{ + if (ses->serv->dhcpv4_relay) + dhcpv4_relay_send(ses->serv->dhcpv4_relay, ses->dhcpv4_request, ses->relay_server_id); + else + __ipoe_session_activate(ses); +} + +static void ipoe_session_keepalive(struct dhcpv4_packet *pack) { + struct ipoe_session *ses = container_of(triton_context_self(), typeof(*ses), ctx); + + if (ses->dhcpv4_request) + dhcpv4_packet_free(ses->dhcpv4_request); + + ses->dhcpv4_request = pack; + if (ses->timer.tpd) triton_timer_mod(&ses->timer, 0); ses->xid = ses->dhcpv4_request->hdr->xid; + + if (ses->ses.state == AP_STATE_ACTIVE && ses->serv->dhcpv4_relay) { + dhcpv4_relay_send(ses->serv->dhcpv4_relay, ses->dhcpv4_request, ses->relay_server_id); + return; + } - if (ses->ses.state == AP_STATE_ACTIVE) - dhcpv4_send_reply(DHCPACK, ses->serv->dhcpv4, ses->dhcpv4_request, ses->yiaddr, ses->siaddr, ses->mask, conf_lease_time); - else + if (ses->ses.state == AP_STATE_ACTIVE) { + dhcpv4_send_reply(DHCPACK, ses->serv->dhcpv4, ses->dhcpv4_request, ses->yiaddr, ses->siaddr, ses->mask, ses->lease_time, ses->dhcpv4_relay_reply); + } else dhcpv4_send_nak(ses->serv->dhcpv4, ses->dhcpv4_request); dhcpv4_packet_free(ses->dhcpv4_request); @@ -520,6 +578,9 @@ static void ipoe_session_finished(struct ap_session *s) if (ses->dhcp_addr) dhcpv4_put_ip(ses->serv->dhcpv4, ses->yiaddr); + + if (ses->relay_addr && ses->serv->dhcpv4_relay) + dhcpv4_relay_send_release(ses->serv->dhcpv4_relay, ses->hwaddr, ses->xid, ses->yiaddr, ses->client_id, ses->relay_agent); if (ses->ifcfg) ipoe_ifcfg_del(ses); @@ -534,7 +595,7 @@ static void ipoe_session_terminate(struct ap_session *s, int hard) { struct ipoe_session *ses = container_of(s, typeof(*ses), ses); - if (ses->l4_redirect) + if (ses->l4_redirect_set) ipoe_change_l4_redirect(ses, 1); ap_session_finished(s); @@ -556,7 +617,7 @@ static struct ipoe_session *ipoe_session_create_dhcpv4(struct ipoe_serv *serv, s struct ipoe_session *ses; int dlen = 0; uint8_t *ptr; - + ses = mempool_alloc(ses_pool); if (!ses) { log_emerg("out of memery\n"); @@ -574,15 +635,13 @@ static struct ipoe_session *ipoe_session_create_dhcpv4(struct ipoe_serv *serv, s ses->xid = pack->hdr->xid; memcpy(ses->hwaddr, pack->hdr->chaddr, 6); ses->giaddr = pack->hdr->giaddr; + ses->lease_time = conf_lease_time; - if (pack->agent_circuit_id) - dlen += sizeof(struct dhcp_opt) + pack->agent_circuit_id->len; - - if (pack->agent_remote_id) - dlen += sizeof(struct dhcp_opt) + pack->agent_remote_id->len; - if (pack->client_id) - dlen += sizeof(struct dhcp_opt) + pack->client_id->len; + dlen += sizeof(struct dhcpv4_option) + pack->client_id->len; + + if (pack->relay_agent) + dlen += sizeof(struct dhcpv4_option) + pack->relay_agent->len; if (dlen) { ses->data = _malloc(dlen); @@ -594,25 +653,20 @@ static struct ipoe_session *ipoe_session_create_dhcpv4(struct ipoe_serv *serv, s ptr = ses->data; } - if (pack->agent_circuit_id) { - ses->agent_circuit_id = (struct dhcp_opt *)ptr; - ses->agent_circuit_id->len = pack->agent_circuit_id->len; - memcpy(ses->agent_circuit_id->data, pack->agent_circuit_id->data, pack->agent_circuit_id->len); - ptr += sizeof(struct dhcp_opt) + pack->agent_circuit_id->len; - } - - if (pack->agent_remote_id) { - ses->agent_remote_id = (struct dhcp_opt *)ptr; - ses->agent_remote_id->len = pack->agent_remote_id->len; - memcpy(ses->agent_remote_id->data, pack->agent_remote_id->data, pack->agent_remote_id->len); - ptr += sizeof(struct dhcp_opt) + pack->agent_remote_id->len; - } - if (pack->client_id) { - ses->client_id = (struct dhcp_opt *)ptr; + ses->client_id = (struct dhcpv4_option *)ptr; ses->client_id->len = pack->client_id->len; memcpy(ses->client_id->data, pack->client_id->data, pack->client_id->len); - ptr += sizeof(struct dhcp_opt) + pack->client_id->len; + ptr += sizeof(struct dhcpv4_option) + pack->client_id->len; + } + + if (pack->relay_agent) { + ses->relay_agent = (struct dhcpv4_option *)ptr; + ses->relay_agent->len = pack->relay_agent->len; + memcpy(ses->relay_agent->data, pack->relay_agent->data, pack->relay_agent->len); + ptr += sizeof(struct dhcpv4_option) + pack->relay_agent->len; + if (dhcpv4_parse_opt82(ses->relay_agent, &ses->agent_circuit_id, &ses->agent_remote_id)) + ses->relay_agent = NULL; } ses->ctx.before_switch = log_switch; @@ -661,24 +715,25 @@ static void ipoe_recv_dhcpv4(struct dhcpv4_serv *dhcpv4, struct dhcpv4_packet *p ses = ipoe_session_lookup(serv, pack); if (!ses) { ses = ipoe_session_create_dhcpv4(serv, pack); + if (ses) { + dhcpv4_packet_ref(pack); - if (conf_verbose && ses) { - log_switch(dhcpv4->ctx, &ses->ses); - log_ppp_info2("recv "); - dhcpv4_print_packet(pack, log_ppp_info2); + if (conf_verbose) { + log_switch(dhcpv4->ctx, &ses->ses); + log_ppp_info2("recv "); + dhcpv4_print_packet(pack, 0, log_ppp_info2); + } } } else { log_switch(dhcpv4->ctx, &ses->ses); if (conf_verbose) { log_ppp_info2("recv "); - dhcpv4_print_packet(pack, log_ppp_info2); + dhcpv4_print_packet(pack, 0, log_ppp_info2); } - if (ses->ses.ipv4 && ses->ses.state == AP_STATE_ACTIVE && pack->request_ip == ses->ses.ipv4->peer_addr) - dhcpv4_send_reply(DHCPOFFER, dhcpv4, pack, ses->yiaddr, ses->siaddr, ses->mask, conf_lease_time); - - dhcpv4_packet_free(pack); + if (ses->ses.state == AP_STATE_ACTIVE && pack->request_ip == ses->yiaddr) + dhcpv4_send_reply(DHCPOFFER, dhcpv4, pack, ses->yiaddr, ses->siaddr, ses->mask, ses->lease_time, ses->dhcpv4_relay_reply); } } else if (pack->msg_type == DHCPREQUEST) { ses = ipoe_session_lookup(serv, pack); @@ -686,7 +741,7 @@ static void ipoe_recv_dhcpv4(struct dhcpv4_serv *dhcpv4, struct dhcpv4_packet *p if (!ses) { if (conf_verbose) { log_info2("recv "); - dhcpv4_print_packet(pack, log_info2); + dhcpv4_print_packet(pack, 0, log_info2); } dhcpv4_send_nak(dhcpv4, pack); @@ -696,52 +751,146 @@ static void ipoe_recv_dhcpv4(struct dhcpv4_serv *dhcpv4, struct dhcpv4_packet *p if (conf_verbose) { log_info2("recv "); - dhcpv4_print_packet(pack, log_info2); + dhcpv4_print_packet(pack, 0, log_info2); } - if (pack->server_id == ses->siaddr && pack->request_ip && pack->request_ip != ses->yiaddr) + if (pack->server_id == ses->siaddr) dhcpv4_send_nak(dhcpv4, pack); + else if (ses->serv->dhcpv4_relay) + dhcpv4_relay_send(ses->serv->dhcpv4_relay, pack, 0); ap_session_terminate(&ses->ses, TERM_USER_REQUEST, 0); } else { if (conf_verbose) { log_switch(dhcpv4->ctx, &ses->ses); log_ppp_info2("recv "); - dhcpv4_print_packet(pack, log_ppp_info2); + dhcpv4_print_packet(pack, 0, log_ppp_info2); } if (serv->opt_shared == 0) ipoe_drop_sessions(serv, ses); - if (ses->ses.state == AP_STATE_STARTING && !ses->dhcpv4_request) { + if (ses->ses.state == AP_STATE_STARTING && ses->yiaddr && !ses->dhcpv4_request) { ses->dhcpv4_request = pack; - pack = NULL; + dhcpv4_packet_ref(pack); triton_context_call(&ses->ctx, (triton_event_func)ipoe_session_activate, ses); - } else if (ses->ses.state == AP_STATE_ACTIVE && !ses->dhcpv4_request) { - ses->dhcpv4_request = pack; - pack = NULL; - triton_context_call(&ses->ctx, (triton_event_func)ipoe_session_keepalive, ses); + } else if (ses->ses.state == AP_STATE_ACTIVE) { + dhcpv4_packet_ref(pack); + triton_context_call(&ses->ctx, (triton_event_func)ipoe_session_keepalive, pack); } } } - if (pack) - dhcpv4_packet_free(pack); } else if (pack->msg_type == DHCPDECLINE || pack->msg_type == DHCPRELEASE) { ses = ipoe_session_lookup(serv, pack); if (ses) { if (conf_verbose) { log_switch(dhcpv4->ctx, &ses->ses); log_ppp_info2("recv "); - dhcpv4_print_packet(pack, log_ppp_info2); + dhcpv4_print_packet(pack, 0, log_ppp_info2); } + + if (pack->msg_type == DHCPDECLINE && ses->serv->dhcpv4_relay) + dhcpv4_relay_send(ses->serv->dhcpv4_relay, pack, 0); ap_session_terminate(&ses->ses, TERM_USER_REQUEST, 0); } + } + pthread_mutex_unlock(&serv->lock); +} + +static int parse_dhcpv4_mask(uint32_t mask) +{ + int i; + + for (i = 31; i >= 0 && (mask & (1 << i)); i--); + + return 32 - (i + 1); +} + +static void ipoe_ses_recv_dhcpv4_relay(struct ipoe_session *ses) +{ + struct dhcpv4_packet *pack = ses->dhcpv4_relay_reply; + struct dhcpv4_option *opt; + + if (conf_verbose) { + log_ppp_info2("recv "); + dhcpv4_print_packet(pack, 1, log_ppp_info2); + } + + if (pack->msg_type == DHCPOFFER && ses->ses.state == AP_STATE_STARTING) { + triton_timer_del(&ses->timer); + + ses->relay_server_id = pack->server_id; + + if (!ses->yiaddr) { + ses->yiaddr = pack->hdr->yiaddr; + ses->relay_addr = 1; + } + + if (!ses->siaddr) { + opt = dhcpv4_packet_find_opt(pack, 3); + if (opt) + ses->siaddr = *(in_addr_t *)opt->data; + } + + opt = dhcpv4_packet_find_opt(pack, 51); + if (opt) + ses->lease_time = ntohl(*(uint32_t *)opt->data); + + opt = dhcpv4_packet_find_opt(pack, 1); + if (opt) + ses->mask = parse_dhcpv4_mask(ntohl(*(uint32_t *)opt->data)); + + __ipoe_session_start(ses); + } else if (pack->msg_type == DHCPACK) { + if (ses->ses.state == AP_STATE_STARTING) + __ipoe_session_activate(ses); + else + dhcpv4_send_reply(DHCPACK, ses->serv->dhcpv4, ses->dhcpv4_request, ses->yiaddr, ses->siaddr, ses->mask, ses->lease_time, ses->dhcpv4_relay_reply); + + } else if (pack->msg_type == DHCPNAK) { + dhcpv4_send_nak(ses->serv->dhcpv4, ses->dhcpv4_request); + ap_session_terminate(&ses->ses, TERM_NAS_REQUEST, 0); + return; + } + + dhcpv4_packet_free(ses->dhcpv4_relay_reply); + ses->dhcpv4_relay_reply = NULL; +} + +static void ipoe_recv_dhcpv4_relay(struct dhcpv4_packet *pack) +{ + struct ipoe_serv *serv = container_of(triton_context_self(), typeof(*serv), ctx); + struct ipoe_session *ses; + int found = 0; + //struct dhcpv4_packet *reply; + + if (ap_shutdown) { dhcpv4_packet_free(pack); + return; + } + + pthread_mutex_lock(&serv->lock); + list_for_each_entry(ses, &serv->sessions, entry) { + if (ses->xid != pack->hdr->xid) + continue; + if (memcmp(ses->hwaddr, pack->hdr->chaddr, 6)) + continue; + + found = 1; + break; } + + if (found && !ses->dhcpv4_relay_reply) { + ses->dhcpv4_relay_reply = pack; + triton_context_call(&ses->ctx, (triton_event_func)ipoe_ses_recv_dhcpv4_relay, ses); + } else + dhcpv4_packet_free(pack); + pthread_mutex_unlock(&serv->lock); } + static struct ipoe_session *ipoe_session_create_up(struct ipoe_serv *serv, struct ethhdr *eth, struct iphdr *iph) { struct ipoe_session *ses; @@ -908,7 +1057,7 @@ static void ev_radius_coa(struct ev_radius_t *ev) } //if (l4_redirect && !ses->l4_redirect) || (!l4_redirect && ses->l4_redirect)) - if (l4_redirect != ses->l4_redirect) + if (l4_redirect != ses->l4_redirect && ev->ses->state == AP_STATE_ACTIVE) ipoe_change_l4_redirect(ses, l4_redirect); } #endif @@ -927,6 +1076,11 @@ static void ipoe_serv_close(struct triton_context_t *ctx) if (serv->dhcpv4) dhcpv4_free(serv->dhcpv4); + + if (serv->dhcpv4_relay) { + ipoe_serv_del_addr(serv, serv->dhcpv4_relay->giaddr); + dhcpv4_relay_free(serv->dhcpv4_relay, &serv->ctx); + } triton_context_unregister(ctx); @@ -983,7 +1137,7 @@ struct ipoe_serv *ipoe_find_serv(const char *ifname) static void add_interface(const char *ifname, int ifindex, const char *opt) { - char *str0, *str, *ptr1, *ptr2; + char *str0 = NULL, *str, *ptr1, *ptr2; int end; struct ipoe_serv *serv; int opt_shared = conf_shared; @@ -991,6 +1145,10 @@ static void add_interface(const char *ifname, int ifindex, const char *opt) int opt_up = 0; int opt_mode = conf_mode; int opt_ifcfg = conf_ifcfg; + const char *opt_relay = conf_relay; + const char *opt_giaddr = NULL; + in_addr_t relay_addr = 0; + in_addr_t giaddr = 0; str0 = strchr(opt, ','); if (str0) { @@ -1033,6 +1191,12 @@ static void add_interface(const char *ifname, int ifindex, const char *opt) goto parse_err; } else if (strcmp(str, "ifcfg") == 0) { opt_ifcfg = atoi(ptr1); + } else if (strcmp(str, "relay") == 0) { + opt_relay = ptr1; + relay_addr = inet_addr(ptr1); + } else if (strcmp(str, "giaddr") == 0) { + opt_giaddr = ptr1; + giaddr = inet_addr(ptr1); } if (end) @@ -1040,9 +1204,7 @@ static void add_interface(const char *ifname, int ifindex, const char *opt) str = ptr2 + 1; } - - _free(str0); - } + } if (!opt_up && !opt_dhcpv4) { opt_up = conf_up; @@ -1074,6 +1236,20 @@ static void add_interface(const char *ifname, int ifindex, const char *opt) serv->opt_mode = opt_mode; serv->opt_ifcfg = opt_ifcfg; + if (serv->dhcpv4_relay && + (serv->dhcpv4_relay->addr != relay_addr || serv->dhcpv4_relay->giaddr != giaddr)) { + ipoe_serv_del_addr(serv, serv->dhcpv4_relay->giaddr); + dhcpv4_relay_free(serv->dhcpv4_relay, &serv->ctx); + serv->dhcpv4_relay = NULL; + } + + if (serv->opt_dhcpv4 && opt_relay && opt_giaddr) + ipoe_serv_add_addr(serv, serv->dhcpv4_relay->giaddr); + serv->dhcpv4_relay = dhcpv4_relay_create(opt_relay, opt_giaddr, &serv->ctx, (triton_event_func)ipoe_recv_dhcpv4_relay); + + if (str0) + _free(str0); + return; } @@ -1098,12 +1274,20 @@ static void add_interface(const char *ifname, int ifindex, const char *opt) serv->dhcpv4 = dhcpv4_create(&serv->ctx, serv->ifname, opt); if (serv->dhcpv4) serv->dhcpv4->recv = ipoe_recv_dhcpv4; + + if (opt_relay && opt_giaddr) { + ipoe_serv_add_addr(serv, giaddr); + serv->dhcpv4_relay = dhcpv4_relay_create(opt_relay, opt_giaddr, &serv->ctx, (triton_event_func)ipoe_recv_dhcpv4_relay); + } } triton_context_wakeup(&serv->ctx); list_add_tail(&serv->entry, &serv_list); + if (str0) + _free(str0); + return; parse_err: @@ -1274,6 +1458,7 @@ static void parse_conf_rad_attr(const char *opt, int *val) } else *val = -1; } + static void load_radius_attrs(void) { parse_conf_rad_attr("attr-dhcp-client-ip", &conf_attr_dhcp_client_ip); @@ -1330,7 +1515,7 @@ static void load_config(void) if (opt) conf_lease_time = atoi(opt); - opt = conf_get_opt("ipoe", "lease-timeout"); + opt = conf_get_opt("ipoe", "max-lease-time"); if (opt) conf_lease_timeout = atoi(opt); @@ -1366,6 +1551,8 @@ static void load_config(void) log_emerg("ipoe: failed to parse 'mode=%s'\n", opt); } else conf_mode = MODE_L2; + + conf_relay = conf_get_opt("ipoe", "relay"); conf_dhcpv4 = 0; conf_up = 0; @@ -1403,9 +1590,10 @@ static void ipoe_init(void) triton_event_register_handler(EV_CONFIG_RELOAD, (triton_event_func)load_config); #ifdef RADIUS - if (triton_module_loaded("radius")) + /*if (triton_module_loaded("radius")) { triton_event_register_handler(EV_RADIUS_ACCESS_ACCEPT, (triton_event_func)ev_radius_access_accept); triton_event_register_handler(EV_RADIUS_COA, (triton_event_func)ev_radius_coa); + }*/ #endif } diff --git a/accel-pppd/ctrl/ipoe/ipoe.h b/accel-pppd/ctrl/ipoe/ipoe.h index b955b95..2d99ca6 100644 --- a/accel-pppd/ctrl/ipoe/ipoe.h +++ b/accel-pppd/ctrl/ipoe/ipoe.h @@ -18,6 +18,7 @@ struct ipoe_serv struct list_head sessions; struct list_head addr_list; struct dhcpv4_serv *dhcpv4; + struct dhcpv4_relay *dhcpv4_relay; pthread_mutex_t lock; int opt_mode; int opt_shared:1; @@ -27,12 +28,6 @@ struct ipoe_serv int need_close:1; }; -struct dhcp_opt -{ - uint8_t len; - uint8_t data[0]; -}; - struct ipoe_session { struct list_head entry; @@ -42,20 +37,26 @@ struct ipoe_session struct ap_ctrl ctrl; struct ap_session ses; uint8_t hwaddr[6]; - struct dhcp_opt *client_id; - struct dhcp_opt *agent_circuit_id; - struct dhcp_opt *agent_remote_id; + struct dhcpv4_option *client_id; + struct dhcpv4_option *relay_agent; + uint8_t *agent_circuit_id; + uint8_t *agent_remote_id; uint32_t xid; uint32_t giaddr; uint32_t yiaddr; uint32_t siaddr; + uint32_t relay_server_id; int mask; + int lease_time; uint8_t *data; struct dhcpv4_packet *dhcpv4_request; + struct dhcpv4_packet *dhcpv4_relay_reply; int ifindex; int ifcfg:1; int dhcp_addr:1; + int relay_addr:1; int l4_redirect:1; + int l4_redirect_set:1; }; struct ipoe_session_info diff --git a/accel-pppd/ctrl/ipoe/lua.c b/accel-pppd/ctrl/ipoe/lua.c index 0b24635..dbc87e2 100644 --- a/accel-pppd/ctrl/ipoe/lua.c +++ b/accel-pppd/ctrl/ipoe/lua.c @@ -146,7 +146,7 @@ static int packet4_agent_circuit_id(lua_State *L) return 0; if (ses->agent_circuit_id) - lua_pushlstring(L, (char *)ses->agent_circuit_id->data, ses->agent_circuit_id->len); + lua_pushlstring(L, (char *)(ses->agent_circuit_id + 1), *ses->agent_circuit_id); else lua_pushnil(L); @@ -161,7 +161,7 @@ static int packet4_agent_remote_id(lua_State *L) return 0; if (ses->agent_remote_id) - lua_pushlstring(L, (char *)ses->agent_remote_id->data, ses->agent_remote_id->len); + lua_pushlstring(L, (char *)(ses->agent_remote_id + 1), *ses->agent_remote_id); else lua_pushnil(L); |