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