1d4afb5ceSopenharmony_ci /* 2d4afb5ceSopenharmony_ci * libwebsockets - small server side websockets and web server implementation 3d4afb5ceSopenharmony_ci * 4d4afb5ceSopenharmony_ci * Copyright (C) 2010 - 2021 Andy Green <andy@warmcat.com> 5d4afb5ceSopenharmony_ci * 6d4afb5ceSopenharmony_ci * Permission is hereby granted, free of charge, to any person obtaining a copy 7d4afb5ceSopenharmony_ci * of this software and associated documentation files (the "Software"), to 8d4afb5ceSopenharmony_ci * deal in the Software without restriction, including without limitation the 9d4afb5ceSopenharmony_ci * rights to use, copy, modify, merge, publish, distribute, sublicense, and/or 10d4afb5ceSopenharmony_ci * sell copies of the Software, and to permit persons to whom the Software is 11d4afb5ceSopenharmony_ci * furnished to do so, subject to the following conditions: 12d4afb5ceSopenharmony_ci * 13d4afb5ceSopenharmony_ci * The above copyright notice and this permission notice shall be included in 14d4afb5ceSopenharmony_ci * all copies or substantial portions of the Software. 15d4afb5ceSopenharmony_ci * 16d4afb5ceSopenharmony_ci * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR 17d4afb5ceSopenharmony_ci * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, 18d4afb5ceSopenharmony_ci * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE 19d4afb5ceSopenharmony_ci * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER 20d4afb5ceSopenharmony_ci * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 21d4afb5ceSopenharmony_ci * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS 22d4afb5ceSopenharmony_ci * IN THE SOFTWARE. 23d4afb5ceSopenharmony_ci */ 24d4afb5ceSopenharmony_ci 25d4afb5ceSopenharmony_ci#include "private-lib-core.h" 26d4afb5ceSopenharmony_ci#include "private-lib-system-dhcpclient.h" 27d4afb5ceSopenharmony_ci 28d4afb5ceSopenharmony_civoid 29d4afb5ceSopenharmony_cilws_dhcpc_retry_write(struct lws_sorted_usec_list *sul) 30d4afb5ceSopenharmony_ci{ 31d4afb5ceSopenharmony_ci lws_dhcpc_req_t *r = lws_container_of(sul, lws_dhcpc_req_t, sul_write); 32d4afb5ceSopenharmony_ci 33d4afb5ceSopenharmony_ci lwsl_debug("%s\n", __func__); 34d4afb5ceSopenharmony_ci 35d4afb5ceSopenharmony_ci if (r && r->wsi_raw) 36d4afb5ceSopenharmony_ci lws_callback_on_writable(r->wsi_raw); 37d4afb5ceSopenharmony_ci} 38d4afb5ceSopenharmony_ci 39d4afb5ceSopenharmony_cistatic void 40d4afb5ceSopenharmony_cilws_dhcpc_destroy(lws_dhcpc_req_t **pr) 41d4afb5ceSopenharmony_ci{ 42d4afb5ceSopenharmony_ci lws_dhcpc_req_t *r = *pr; 43d4afb5ceSopenharmony_ci 44d4afb5ceSopenharmony_ci lws_sul_cancel(&r->sul_conn); 45d4afb5ceSopenharmony_ci lws_sul_cancel(&r->sul_write); 46d4afb5ceSopenharmony_ci lws_sul_cancel(&r->sul_renew); 47d4afb5ceSopenharmony_ci 48d4afb5ceSopenharmony_ci if (r->wsi_raw) 49d4afb5ceSopenharmony_ci lws_set_timeout(r->wsi_raw, 1, LWS_TO_KILL_ASYNC); 50d4afb5ceSopenharmony_ci 51d4afb5ceSopenharmony_ci lws_dll2_remove(&r->list); 52d4afb5ceSopenharmony_ci 53d4afb5ceSopenharmony_ci lws_free_set_NULL(r); 54d4afb5ceSopenharmony_ci} 55d4afb5ceSopenharmony_ci 56d4afb5ceSopenharmony_ciint 57d4afb5ceSopenharmony_cilws_dhcpc_status(struct lws_context *context, lws_sockaddr46 *sa46) 58d4afb5ceSopenharmony_ci{ 59d4afb5ceSopenharmony_ci lws_dhcpc_req_t *r; 60d4afb5ceSopenharmony_ci 61d4afb5ceSopenharmony_ci lws_start_foreach_dll(struct lws_dll2 *, p, context->dhcpc_owner.head) { 62d4afb5ceSopenharmony_ci r = (lws_dhcpc_req_t *)p; 63d4afb5ceSopenharmony_ci 64d4afb5ceSopenharmony_ci if (r->state == LDHC_BOUND) { 65d4afb5ceSopenharmony_ci if (sa46) { 66d4afb5ceSopenharmony_ci memcpy(sa46, &r->is.sa46[LWSDH_SA46_DNS_SRV_1], 67d4afb5ceSopenharmony_ci sizeof(*sa46)); 68d4afb5ceSopenharmony_ci } 69d4afb5ceSopenharmony_ci return 1; 70d4afb5ceSopenharmony_ci } 71d4afb5ceSopenharmony_ci 72d4afb5ceSopenharmony_ci } lws_end_foreach_dll(p); 73d4afb5ceSopenharmony_ci 74d4afb5ceSopenharmony_ci return 0; 75d4afb5ceSopenharmony_ci} 76d4afb5ceSopenharmony_ci 77d4afb5ceSopenharmony_cistatic lws_dhcpc_req_t * 78d4afb5ceSopenharmony_cilws_dhcpc_find(struct lws_context *context, const char *iface, int af) 79d4afb5ceSopenharmony_ci{ 80d4afb5ceSopenharmony_ci lws_dhcpc_req_t *r; 81d4afb5ceSopenharmony_ci 82d4afb5ceSopenharmony_ci /* see if we are already looking after this af / iface combination */ 83d4afb5ceSopenharmony_ci 84d4afb5ceSopenharmony_ci lws_start_foreach_dll(struct lws_dll2 *, p, context->dhcpc_owner.head) { 85d4afb5ceSopenharmony_ci r = (lws_dhcpc_req_t *)p; 86d4afb5ceSopenharmony_ci 87d4afb5ceSopenharmony_ci if (!strcmp((const char *)&r[1], iface) && af == r->af) 88d4afb5ceSopenharmony_ci return r; /* yes... */ 89d4afb5ceSopenharmony_ci 90d4afb5ceSopenharmony_ci } lws_end_foreach_dll(p); 91d4afb5ceSopenharmony_ci 92d4afb5ceSopenharmony_ci return NULL; 93d4afb5ceSopenharmony_ci} 94d4afb5ceSopenharmony_ci 95d4afb5ceSopenharmony_ci/* 96d4afb5ceSopenharmony_ci * Create a persistent dhcp client entry for network interface "iface" and AF 97d4afb5ceSopenharmony_ci * type "af" 98d4afb5ceSopenharmony_ci */ 99d4afb5ceSopenharmony_ci 100d4afb5ceSopenharmony_ciint 101d4afb5ceSopenharmony_cilws_dhcpc_request(struct lws_context *context, const char *iface, int af, 102d4afb5ceSopenharmony_ci dhcpc_cb_t cb, void *opaque) 103d4afb5ceSopenharmony_ci{ 104d4afb5ceSopenharmony_ci lws_dhcpc_req_t *r = lws_dhcpc_find(context, iface, af); 105d4afb5ceSopenharmony_ci int n; 106d4afb5ceSopenharmony_ci 107d4afb5ceSopenharmony_ci /* see if we are already looking after this af / iface combination */ 108d4afb5ceSopenharmony_ci 109d4afb5ceSopenharmony_ci if (r) 110d4afb5ceSopenharmony_ci return 0; 111d4afb5ceSopenharmony_ci 112d4afb5ceSopenharmony_ci /* nope... let's create a request object as he asks */ 113d4afb5ceSopenharmony_ci 114d4afb5ceSopenharmony_ci n = (int)strlen(iface); 115d4afb5ceSopenharmony_ci r = lws_zalloc(sizeof(*r) + (unsigned int)n + 1u, __func__); 116d4afb5ceSopenharmony_ci if (!r) 117d4afb5ceSopenharmony_ci return 1; 118d4afb5ceSopenharmony_ci 119d4afb5ceSopenharmony_ci memcpy(&r[1], iface, (unsigned int)n + 1); 120d4afb5ceSopenharmony_ci r->af = (uint8_t)af; 121d4afb5ceSopenharmony_ci r->cb = cb; 122d4afb5ceSopenharmony_ci r->opaque = opaque; 123d4afb5ceSopenharmony_ci r->context = context; 124d4afb5ceSopenharmony_ci r->state = LDHC_INIT; 125d4afb5ceSopenharmony_ci 126d4afb5ceSopenharmony_ci lws_strncpy(r->is.ifname, iface, sizeof(r->is.ifname)); 127d4afb5ceSopenharmony_ci 128d4afb5ceSopenharmony_ci lws_dll2_add_head(&r->list, &context->dhcpc_owner); /* add him to list */ 129d4afb5ceSopenharmony_ci 130d4afb5ceSopenharmony_ci lws_dhcpc4_retry_conn(&r->sul_conn); 131d4afb5ceSopenharmony_ci 132d4afb5ceSopenharmony_ci return 0; 133d4afb5ceSopenharmony_ci} 134d4afb5ceSopenharmony_ci 135d4afb5ceSopenharmony_ci/* 136d4afb5ceSopenharmony_ci * Destroy every DHCP client object related to interface "iface" 137d4afb5ceSopenharmony_ci */ 138d4afb5ceSopenharmony_ci 139d4afb5ceSopenharmony_cistatic int 140d4afb5ceSopenharmony_ci_remove_if(struct lws_dll2 *d, void *opaque) 141d4afb5ceSopenharmony_ci{ 142d4afb5ceSopenharmony_ci lws_dhcpc_req_t *r = lws_container_of(d, lws_dhcpc_req_t, list); 143d4afb5ceSopenharmony_ci 144d4afb5ceSopenharmony_ci if (!opaque || !strcmp((const char *)&r[1], (const char *)opaque)) 145d4afb5ceSopenharmony_ci lws_dhcpc_destroy(&r); 146d4afb5ceSopenharmony_ci 147d4afb5ceSopenharmony_ci return 0; 148d4afb5ceSopenharmony_ci} 149d4afb5ceSopenharmony_ci 150d4afb5ceSopenharmony_ciint 151d4afb5ceSopenharmony_cilws_dhcpc_remove(struct lws_context *context, const char *iface) 152d4afb5ceSopenharmony_ci{ 153d4afb5ceSopenharmony_ci lws_dll2_foreach_safe(&context->dhcpc_owner, (void *)iface, _remove_if); 154d4afb5ceSopenharmony_ci 155d4afb5ceSopenharmony_ci return 0; 156d4afb5ceSopenharmony_ci} 157