summaryrefslogtreecommitdiffstats
path: root/tinyNET/src/dhcp/tnet_dhcp.c
diff options
context:
space:
mode:
Diffstat (limited to 'tinyNET/src/dhcp/tnet_dhcp.c')
-rw-r--r--tinyNET/src/dhcp/tnet_dhcp.c338
1 files changed, 338 insertions, 0 deletions
diff --git a/tinyNET/src/dhcp/tnet_dhcp.c b/tinyNET/src/dhcp/tnet_dhcp.c
new file mode 100644
index 0000000..8e7dbe6
--- /dev/null
+++ b/tinyNET/src/dhcp/tnet_dhcp.c
@@ -0,0 +1,338 @@
+/*
+* Copyright (C) 2009-2010 Mamadou Diop.
+*
+* Contact: Mamadou Diop <diopmamadou(at)doubango.org>
+*
+* This file is part of Open Source Doubango Framework.
+*
+* DOUBANGO is free software: you can redistribute it and/or modify
+* it under the terms of the GNU General Public License as published by
+* the Free Software Foundation, either version 3 of the License, or
+* (at your option) any later version.
+*
+* DOUBANGO is distributed in the hope that it will be useful,
+* but WITHOUT ANY WARRANTY; without even the implied warranty of
+* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+* GNU General Public License for more details.
+*
+* You should have received a copy of the GNU General Public License
+* along with DOUBANGO.
+*
+*/
+/**@file tnet_dhcp.c
+ * @brief DHCP/BOOTP (RFC 2131 - Dynamic Host Configuration Protocol) utilities function for P-CSCF discovery(RFC 3319 and 3361).
+ * Also implement: RFC 3315, 3118, 3319, 3825 (Geoconf), 4676 (Civic Addresses Configuration Information) ...
+ *
+ * @author Mamadou Diop <diopmamadou(at)doubango.org>
+ *
+ * @date Created: Sat Nov 8 16:54:58 2009 mdiop
+ */
+#include "tnet_dhcp.h"
+
+#include "../tnet_endianness.h"
+
+#include "tsk_string.h"
+#include "tsk_thread.h"
+#include "tsk_memory.h"
+#include "tsk_time.h"
+#include "tsk_debug.h"
+
+#include <string.h>
+
+// Useful link: http://support.microsoft.com/?scid=kb%3Ben-us%3B169289&x=21&y=14
+// Another one: http://www.iana.org/assignments/bootp-dhcp-parameters/
+// Another one: http://www.slideshare.net/raini/DHCP-Presentation-v102
+// RFC 3319 Dynamic Host Configuration Protocol (DHCPv6) Options for Session Initiation Protocol (SIP) Servers
+// RFC 3361 Dynamic Host Configuration Protocol (DHCP-for-IPv4) Option for Session Initiation Protocol (SIP) Servers
+
+/**@defgroup tnet_dhcp_group DHCPv4/BOOTP (RFC 2131) implementation.
+*/
+
+/**@ingroup tnet_dhcp_group
+* Creates new DHCPv4 context.
+*/
+tnet_dhcp_ctx_t* tnet_dhcp_ctx_create()
+{
+ return tsk_object_new(tnet_dhcp_ctx_def_t);
+}
+
+/**@ingroup tnet_dhcp_group
+* Creates new DHCPv4 parameters.
+*/
+tnet_dhcp_params_t* tnet_dhcp_params_create()
+{
+ return tsk_object_new(tnet_dhcp_params_def_t);
+}
+
+/* FIXME: USE retransmission mech (*2*2...)
+*/
+/**@ingroup tnet_dhcp_group
+*/
+tnet_dhcp_reply_t* tnet_dhcp_send_request(tnet_dhcp_ctx_t* ctx, tnet_dhcp_request_t* request)
+{
+ tsk_buffer_t *output;
+ tnet_dhcp_reply_t* reply = tsk_null;
+ int ret;
+ struct timeval tv;
+ fd_set set;
+ uint64_t timeout = 0;
+ tsk_list_item_t *item;
+ const tnet_interface_t *iface;
+
+ tnet_socket_t *localsocket4 = tsk_null;
+ struct sockaddr_storage server;
+
+ if(!ctx || !request){
+ goto bail;
+ }
+
+ localsocket4 = tnet_socket_create(TNET_SOCKET_HOST_ANY, ctx->port_client, tnet_socket_type_udp_ipv4);
+ if(!TNET_SOCKET_IS_VALID(localsocket4)){
+ TSK_DEBUG_ERROR("Failed to create/bind DHCP client socket.");
+ goto bail;
+ }
+
+ /* Always wait for 200ms before retransmission */
+ tv.tv_sec = 0;
+ tv.tv_usec = (200 * 1000);
+
+ if(tnet_sockaddr_init("255.255.255.255", ctx->server_port, tnet_socket_type_udp_ipv4, &server)){
+ TNET_PRINT_LAST_ERROR("Failed to initialize the DHCP server address");
+ goto bail;
+ }
+
+ /* ENABLE BROADCASTING */
+ {
+#if defined(SOLARIS)
+ char yes = '1';
+#else
+ int yes = 1;
+#endif
+ if(setsockopt(localsocket4->fd, SOL_SOCKET, SO_BROADCAST, (char*)&yes, sizeof(int))){
+ TNET_PRINT_LAST_ERROR("Failed to enable broadcast option");
+ goto bail;
+ }
+ }
+
+ /* Set timeout */
+ timeout = tsk_time_epoch() + ctx->timeout;
+
+ do
+ {
+ /* RFC 2131 - 3.6 Use of DHCP in clients with multiple interfaces
+ A client with multiple network interfaces must use DHCP through each
+ interface independently to obtain configuration information
+ parameters for those separate interfaces.
+ */
+
+ tsk_list_foreach(item, ctx->interfaces){
+ iface = item->data;
+
+ /* Set FD */
+ FD_ZERO(&set);
+ FD_SET(localsocket4->fd, &set);
+
+ /* ciaddr */
+ if(request->type == dhcp_type_inform){
+ struct sockaddr_storage ss;
+ if(!tnet_get_sockaddr(localsocket4->fd, &ss)){
+ uint32_t addr = tnet_htonl_2(&((struct sockaddr_in*)&ss)->sin_addr);
+ memcpy(&request->ciaddr, &addr, 4);
+ }
+ }
+
+ /* chaddr */
+ memset(request->chaddr, 0, sizeof(request->chaddr));
+ request->hlen = iface->mac_address_length > sizeof(request->chaddr) ? sizeof(request->chaddr) : iface->mac_address_length;
+ memcpy(request->chaddr, iface->mac_address, request->hlen);
+
+ /* Serialize and send to the server. */
+ if(!(output = tnet_dhcp_message_serialize(ctx, request))){
+ TSK_DEBUG_ERROR("Failed to serialize the DHCP message.");
+ goto next_iface;
+ }
+ /* Send the request to the DHCP server */
+ if((ret =tnet_sockfd_sendto(localsocket4->fd, (const struct sockaddr*)&server, output->data, output->size))<0){
+ TNET_PRINT_LAST_ERROR("Failed to send DHCP request");
+
+ tsk_thread_sleep(150); // wait 150ms before trying the next iface.
+ goto next_iface;
+ }
+ /* wait for response */
+ if((ret = select(localsocket4->fd+1, &set, NULL, NULL, &tv))<0){ /* Error */
+ TNET_PRINT_LAST_ERROR("select have failed.");
+ tsk_thread_sleep(150); // wait 150ms before trying the next iface.
+ goto next_iface;
+ }
+ else if(ret == 0)
+ { /* timeout ==> do nothing */
+ }
+ else
+ { /* there is data to read */
+ tsk_size_t len = 0;
+ void* data = tsk_null;
+
+ /* Check how how many bytes are pending */
+ if((ret = tnet_ioctlt(localsocket4->fd, FIONREAD, &len))<0){
+ goto next_iface;
+ }
+
+ /* Receive pending data */
+ data = tsk_calloc(len, sizeof(uint8_t));
+ if((ret = tnet_sockfd_recv(localsocket4->fd, data, len, 0))<0){
+ TSK_FREE(data);
+
+ TNET_PRINT_LAST_ERROR("Failed to receive DHCP dgrams.");
+ goto next_iface;
+ }
+
+ /* Parse the incoming response. */
+ reply = tnet_dhcp_message_deserialize(ctx, data, (tsk_size_t)ret);
+ TSK_FREE(data);
+
+ if(reply)
+ { /* response successfuly parsed */
+ if(request->xid != reply->xid)
+ { /* Not same transaction id ==> continue*/
+ TSK_OBJECT_SAFE_FREE(reply);
+ }
+ }
+ }
+
+ next_iface:
+ TSK_OBJECT_SAFE_FREE(output);
+ if(reply){
+ goto bail;
+ }
+ }
+ //break;//FIXME
+ }
+ while(timeout > tsk_time_epoch());
+
+bail:
+ TSK_OBJECT_SAFE_FREE(localsocket4);
+
+ return reply;
+}
+
+/**@ingroup tnet_dhcp_group
+*/
+tnet_dhcp_reply_t* tnet_dhcp_query(tnet_dhcp_ctx_t* ctx, tnet_dhcp_message_type_t type, tnet_dhcp_params_t* params)
+{
+ tnet_dhcp_reply_t* reply = tsk_null;
+ tnet_dhcp_request_t* request = tnet_dhcp_request_create();
+
+ if(!ctx || !params || !request){
+ goto bail;
+ }
+
+ request->type = type;
+ tnet_dhcp_message_add_codes(request, params->codes, params->codes_count);
+
+ reply = tnet_dhcp_send_request(ctx, request);
+
+bail:
+ TSK_OBJECT_SAFE_FREE(request);
+
+ return reply;
+}
+
+/**@ingroup tnet_dhcp_group
+*/
+int tnet_dhcp_params_add_code(tnet_dhcp_params_t* params, tnet_dhcp_option_code_t code)
+{
+ if(params){
+ if(params->codes_count < TNET_DHCP_MAX_CODES){
+ unsigned i;
+ for(i=0; i<params->codes_count; i++){
+ if(params->codes[i] == code){
+ return -3;
+ }
+ }
+ params->codes[params->codes_count++] = code;
+ }
+ else return -2;
+ }
+ return -1;
+}
+
+
+
+//=================================================================================================
+// [[DHCP CONTEXT]] object definition
+//
+static tsk_object_t* tnet_dhcp_ctx_ctor(tsk_object_t * self, va_list * app)
+{
+ tnet_dhcp_ctx_t *ctx = self;
+ if(ctx){
+ tnet_host_t host;
+
+ ctx->vendor_id = tsk_strdup(TNET_DHCP_VENDOR_ID_DEFAULT);
+ if(!tnet_gethostname(&host)){
+ ctx->hostname = tsk_strndup(host, tsk_strlen(host));
+ }
+ ctx->timeout = TNET_DHCP_TIMEOUT_DEFAULT;
+ ctx->max_msg_size = TNET_DHCP_MAX_MSG_SIZE;
+ ctx->port_client = TNET_DHCP_CLIENT_PORT;
+ ctx->server_port = TNET_DHCP_SERVER_PORT;
+ ctx->interfaces = tnet_get_interfaces();
+
+ if(!ctx->interfaces || TSK_LIST_IS_EMPTY(ctx->interfaces)){
+ TSK_DEBUG_ERROR("Failed to retrieve network interfaces.");
+ }
+
+ tsk_safeobj_init(ctx);
+ }
+ return self;
+}
+
+static tsk_object_t* tnet_dhcp_ctx_dtor(tsk_object_t * self)
+{
+ tnet_dhcp_ctx_t *ctx = self;
+ if(ctx){
+ tsk_safeobj_deinit(ctx);
+
+ TSK_FREE(ctx->vendor_id);
+ TSK_FREE(ctx->hostname);
+
+ TSK_OBJECT_SAFE_FREE(ctx->interfaces);
+ }
+ return self;
+}
+
+static const tsk_object_def_t tnet_dhcp_ctx_def_s =
+{
+ sizeof(tnet_dhcp_ctx_t),
+ tnet_dhcp_ctx_ctor,
+ tnet_dhcp_ctx_dtor,
+ tsk_null,
+};
+const tsk_object_def_t *tnet_dhcp_ctx_def_t = &tnet_dhcp_ctx_def_s;
+
+//=================================================================================================
+// [[DHCP PARAMS]] object definition
+//
+static tsk_object_t* tnet_dhcp_params_ctor(tsk_object_t * self, va_list * app)
+{
+ tnet_dhcp_params_t *params = self;
+ if(params){
+ }
+ return self;
+}
+
+static tsk_object_t* tnet_dhcp_params_dtor(tsk_object_t * self)
+{
+ tnet_dhcp_params_t *params = self;
+ if(params){
+ }
+ return self;
+}
+
+static const tsk_object_def_t tnet_dhcp_params_def_s =
+{
+ sizeof(tnet_dhcp_params_t),
+ tnet_dhcp_params_ctor,
+ tnet_dhcp_params_dtor,
+ tsk_null,
+};
+const tsk_object_def_t *tnet_dhcp_params_def_t = &tnet_dhcp_params_def_s;
OpenPOWER on IntegriCloud