mirror of
https://github.com/reactos/reactos.git
synced 2025-01-07 06:45:24 +00:00
9febc82acf
- Use a stop event to terminate the dispatch loop. - Report to the services manager that the service can be stopped or shut-down while it is running. CORE-14390
428 lines
12 KiB
C
428 lines
12 KiB
C
/* $OpenBSD: dispatch.c,v 1.31 2004/09/21 04:07:03 david Exp $ */
|
|
|
|
/*
|
|
* Copyright 2004 Henning Brauer <henning@openbsd.org>
|
|
* Copyright (c) 1995, 1996, 1997, 1998, 1999
|
|
* The Internet Software Consortium. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
* 3. Neither the name of The Internet Software Consortium nor the names
|
|
* of its contributors may be used to endorse or promote products derived
|
|
* from this software without specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE INTERNET SOFTWARE CONSORTIUM AND
|
|
* CONTRIBUTORS ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES,
|
|
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE INTERNET SOFTWARE CONSORTIUM OR
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
|
|
* USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
|
|
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
|
|
* SUCH DAMAGE.
|
|
*
|
|
* This software has been written for the Internet Software Consortium
|
|
* by Ted Lemon <mellon@fugue.com> in cooperation with Vixie
|
|
* Enterprises. To learn more about the Internet Software Consortium,
|
|
* see ``http://www.vix.com/isc''. To learn more about Vixie
|
|
* Enterprises, see ``http://www.vix.com''.
|
|
*/
|
|
|
|
#include <rosdhcp.h>
|
|
|
|
//#include <sys/ioctl.h>
|
|
|
|
//#include <net/if_media.h>
|
|
//#include <ifaddrs.h>
|
|
//#include <poll.h>
|
|
|
|
extern SOCKET DhcpSocket;
|
|
HANDLE AdapterStateChangedEvent = NULL;
|
|
struct protocol *protocols = NULL;
|
|
struct timeout *timeouts = NULL;
|
|
static struct timeout *free_timeouts = NULL;
|
|
void (*bootp_packet_handler)(struct interface_info *,
|
|
struct dhcp_packet *, int, unsigned int,
|
|
struct iaddr, struct hardware *);
|
|
|
|
/*
|
|
* Wait for packets to come in using poll(). When a packet comes in,
|
|
* call receive_packet to receive the packet and possibly strip hardware
|
|
* addressing information from it, and then call through the
|
|
* bootp_packet_handler hook to try to do something with it.
|
|
*/
|
|
void
|
|
dispatch(HANDLE hStopEvent)
|
|
{
|
|
int count, to_msec;
|
|
struct protocol *l;
|
|
time_t howlong, cur_time;
|
|
HANDLE Events[3];
|
|
int EventCount = 2;
|
|
|
|
Events[0] = StartAdapterDiscovery();
|
|
if (!Events[0])
|
|
return;
|
|
|
|
AdapterStateChangedEvent = Events[0];
|
|
|
|
Events[1] = hStopEvent;
|
|
Events[2] = WSA_INVALID_EVENT;
|
|
|
|
ApiLock();
|
|
|
|
do {
|
|
/*
|
|
* Call any expired timeouts, and then if there's still
|
|
* a timeout registered, time out the select call then.
|
|
*/
|
|
time(&cur_time);
|
|
|
|
if (timeouts)
|
|
{
|
|
struct timeout *t;
|
|
|
|
if (timeouts->when <= cur_time) {
|
|
t = timeouts;
|
|
timeouts = timeouts->next;
|
|
(*(t->func))(t->what);
|
|
t->next = free_timeouts;
|
|
free_timeouts = t;
|
|
continue;
|
|
}
|
|
|
|
/*
|
|
* Figure timeout in milliseconds, and check for
|
|
* potential overflow, so we can cram into an
|
|
* int for poll, while not polling with a
|
|
* negative timeout and blocking indefinitely.
|
|
*/
|
|
howlong = timeouts->when - cur_time;
|
|
if (howlong > INT_MAX / 1000)
|
|
howlong = INT_MAX / 1000;
|
|
to_msec = howlong * 1000;
|
|
}
|
|
else
|
|
{
|
|
to_msec = INFINITE;
|
|
}
|
|
|
|
if (Events[2] == WSA_INVALID_EVENT && DhcpSocket != INVALID_SOCKET)
|
|
{
|
|
Events[2] = WSACreateEvent();
|
|
if (Events[2] != WSA_INVALID_EVENT)
|
|
{
|
|
count = WSAEventSelect(DhcpSocket, Events[2], FD_READ | FD_CLOSE);
|
|
if (count != NO_ERROR)
|
|
{
|
|
WSACloseEvent(Events[2]);
|
|
Events[2] = WSA_INVALID_EVENT;
|
|
}
|
|
else
|
|
{
|
|
EventCount = 3;
|
|
}
|
|
}
|
|
}
|
|
else if (Events[2] != WSA_INVALID_EVENT && DhcpSocket == INVALID_SOCKET)
|
|
{
|
|
WSACloseEvent(Events[2]);
|
|
Events[2] = WSA_INVALID_EVENT;
|
|
|
|
EventCount = 2;
|
|
}
|
|
|
|
ApiUnlock();
|
|
count = WaitForMultipleObjects(EventCount,
|
|
Events,
|
|
FALSE,
|
|
to_msec);
|
|
ApiLock();
|
|
if (count == WAIT_OBJECT_0)
|
|
{
|
|
/* Adapter state change */
|
|
continue;
|
|
}
|
|
else if (count == WAIT_OBJECT_0 + 1)
|
|
{
|
|
/* Stop event signalled */
|
|
break;
|
|
}
|
|
else if (count == WAIT_OBJECT_0 + 2)
|
|
{
|
|
/* Packet received */
|
|
|
|
/* WSA events are manual reset events */
|
|
WSAResetEvent(Events[2]);
|
|
}
|
|
else
|
|
{
|
|
/* Timeout */
|
|
continue;
|
|
}
|
|
|
|
for (l = protocols; l; l = l->next) {
|
|
struct interface_info *ip;
|
|
ip = l->local;
|
|
if (ip && (l->handler != got_one ||
|
|
!ip->dead)) {
|
|
DH_DbgPrint(MID_TRACE,("Handling %x\n", l));
|
|
(*(l->handler))(l);
|
|
}
|
|
}
|
|
} while (1);
|
|
|
|
AdapterStateChangedEvent = NULL;
|
|
CloseHandle(Events[0]);
|
|
CloseHandle(Events[1]);
|
|
WSACloseEvent(Events[2]);
|
|
|
|
ApiUnlock();
|
|
}
|
|
|
|
void
|
|
got_one(struct protocol *l)
|
|
{
|
|
struct sockaddr_in from;
|
|
struct hardware hfrom;
|
|
struct iaddr ifrom;
|
|
ssize_t result;
|
|
union {
|
|
/*
|
|
* Packet input buffer. Must be as large as largest
|
|
* possible MTU.
|
|
*/
|
|
unsigned char packbuf[4095];
|
|
struct dhcp_packet packet;
|
|
} u;
|
|
struct interface_info *ip = l->local;
|
|
PDHCP_ADAPTER adapter;
|
|
|
|
if ((result = receive_packet(ip, u.packbuf, sizeof(u), &from,
|
|
&hfrom)) == -1) {
|
|
warning("receive_packet failed on %s: %d", ip->name,
|
|
WSAGetLastError());
|
|
ip->errors++;
|
|
if (ip->errors > 20) {
|
|
/* our interface has gone away. */
|
|
warning("Interface %s no longer appears valid.",
|
|
ip->name);
|
|
ip->dead = 1;
|
|
closesocket(l->fd);
|
|
remove_protocol(l);
|
|
adapter = AdapterFindInfo(ip);
|
|
if (adapter) {
|
|
RemoveEntryList(&adapter->ListEntry);
|
|
free(adapter);
|
|
}
|
|
}
|
|
return;
|
|
}
|
|
if (result == 0)
|
|
return;
|
|
|
|
if (bootp_packet_handler) {
|
|
ifrom.len = 4;
|
|
memcpy(ifrom.iabuf, &from.sin_addr, ifrom.len);
|
|
|
|
|
|
adapter = AdapterFindByHardwareAddress(u.packet.chaddr,
|
|
u.packet.hlen);
|
|
|
|
if (!adapter) {
|
|
warning("Discarding packet with a non-matching target physical address\n");
|
|
return;
|
|
}
|
|
|
|
(*bootp_packet_handler)(&adapter->DhclientInfo, &u.packet, result,
|
|
from.sin_port, ifrom, &hfrom);
|
|
}
|
|
}
|
|
|
|
void
|
|
add_timeout(time_t when, void (*where)(void *), void *what)
|
|
{
|
|
struct timeout *t, *q;
|
|
|
|
DH_DbgPrint(MID_TRACE,("Adding timeout %x %p %x\n", when, where, what));
|
|
/* See if this timeout supersedes an existing timeout. */
|
|
t = NULL;
|
|
for (q = timeouts; q; q = q->next) {
|
|
if (q->func == where && q->what == what) {
|
|
if (t)
|
|
t->next = q->next;
|
|
else
|
|
timeouts = q->next;
|
|
break;
|
|
}
|
|
t = q;
|
|
}
|
|
|
|
/* If we didn't supersede a timeout, allocate a timeout
|
|
structure now. */
|
|
if (!q) {
|
|
if (free_timeouts) {
|
|
q = free_timeouts;
|
|
free_timeouts = q->next;
|
|
q->func = where;
|
|
q->what = what;
|
|
} else {
|
|
q = malloc(sizeof(struct timeout));
|
|
if (!q) {
|
|
error("Can't allocate timeout structure!");
|
|
return;
|
|
}
|
|
q->func = where;
|
|
q->what = what;
|
|
}
|
|
}
|
|
|
|
q->when = when;
|
|
|
|
/* Now sort this timeout into the timeout list. */
|
|
|
|
/* Beginning of list? */
|
|
if (!timeouts || timeouts->when > q->when) {
|
|
q->next = timeouts;
|
|
timeouts = q;
|
|
return;
|
|
}
|
|
|
|
/* Middle of list? */
|
|
for (t = timeouts; t->next; t = t->next) {
|
|
if (t->next->when > q->when) {
|
|
q->next = t->next;
|
|
t->next = q;
|
|
return;
|
|
}
|
|
}
|
|
|
|
/* End of list. */
|
|
t->next = q;
|
|
q->next = NULL;
|
|
}
|
|
|
|
void
|
|
cancel_timeout(void (*where)(void *), void *what)
|
|
{
|
|
struct timeout *t, *q;
|
|
|
|
/* Look for this timeout on the list, and unlink it if we find it. */
|
|
t = NULL;
|
|
for (q = timeouts; q; q = q->next) {
|
|
if (q->func == where && q->what == what) {
|
|
if (t)
|
|
t->next = q->next;
|
|
else
|
|
timeouts = q->next;
|
|
break;
|
|
}
|
|
t = q;
|
|
}
|
|
|
|
/* If we found the timeout, put it on the free list. */
|
|
if (q) {
|
|
q->next = free_timeouts;
|
|
free_timeouts = q;
|
|
}
|
|
}
|
|
|
|
/* Add a protocol to the list of protocols... */
|
|
void
|
|
add_protocol(char *name, int fd, void (*handler)(struct protocol *),
|
|
void *local)
|
|
{
|
|
struct protocol *p;
|
|
|
|
p = malloc(sizeof(*p));
|
|
if (!p)
|
|
error("can't allocate protocol struct for %s", name);
|
|
|
|
p->fd = fd;
|
|
p->handler = handler;
|
|
p->local = local;
|
|
p->next = protocols;
|
|
protocols = p;
|
|
}
|
|
|
|
void
|
|
remove_protocol(struct protocol *proto)
|
|
{
|
|
struct protocol *p, *next, *prev;
|
|
struct interface_info *ip = proto->local;
|
|
struct timeout *t, *q, *u;
|
|
|
|
t = NULL;
|
|
q = timeouts;
|
|
while (q != NULL)
|
|
{
|
|
/* Remove all timeouts for this protocol */
|
|
if (q->what == ip)
|
|
{
|
|
/* Unlink the timeout from previous */
|
|
if (t)
|
|
t->next = q->next;
|
|
else
|
|
timeouts = q->next;
|
|
|
|
/* Advance to the next timeout */
|
|
u = q->next;
|
|
|
|
/* Add it to the free list */
|
|
q->next = free_timeouts;
|
|
free_timeouts = q;
|
|
}
|
|
else
|
|
{
|
|
/* Advance to the next timeout */
|
|
u = q->next;
|
|
|
|
/* Update the previous pointer */
|
|
t = q;
|
|
}
|
|
|
|
/* Advance */
|
|
q = u;
|
|
}
|
|
|
|
prev = NULL;
|
|
for (p = protocols; p; p = next) {
|
|
next = p->next;
|
|
if (p == proto) {
|
|
if (prev)
|
|
prev->next = p->next;
|
|
else
|
|
protocols = p->next;
|
|
free(p);
|
|
}
|
|
}
|
|
}
|
|
|
|
struct protocol *
|
|
find_protocol_by_adapter(struct interface_info *info)
|
|
{
|
|
struct protocol *p;
|
|
|
|
for( p = protocols; p; p = p->next ) {
|
|
if( p->local == (void *)info ) return p;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
int
|
|
interface_link_status(char *ifname)
|
|
{
|
|
return (1);
|
|
}
|