/*
Copyright (C) 2017 Free Software Foundation, Inc.
Written by Joan Lledó.
This file is part of the GNU Hurd.
The GNU Hurd 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 2, or (at
your option) any later version.
The GNU Hurd 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 the GNU Hurd. If not, see .
*/
/* Common interface for all kinds of devices */
#include
#include
#include
#include
/* Open the device and set the interface up */
static error_t
if_open (struct netif *netif)
{
error_t err = 0;
struct ifcommon *ifc = netif_get_state (netif);
if (ifc->open)
err = ifc->open (netif);
if (!err)
{
/* Up the inerface */
ifc->flags |= IFF_UP | IFF_RUNNING;
netif_set_up (netif);
}
return err;
}
/* Close the device and set the interface down */
static error_t
if_close (struct netif *netif)
{
error_t err = 0;
struct ifcommon *ifc = netif_get_state (netif);
if (ifc->close)
err = ifc->close (netif);
if (!err)
{
/* Down the inerface */
ifc->flags &= ~(IFF_UP | IFF_RUNNING);
netif_set_down (netif);
}
return err;
}
/*
* Common initialization callback for all kinds of devices.
*
* This function doesn't assume there's a device nor tries to open it.
* If a device is present, it must be opened from the ifc->init() callback.
*/
err_t
if_init (struct netif *netif)
{
struct ifcommon *ifc = netif_get_state (netif);
if (ifc == NULL)
/* The user provided no interface */
return -1;
return ifc->init (netif);
}
/* Tries to close the device and frees allocated resources */
error_t
if_terminate (struct netif * netif)
{
error_t err;
struct ifcommon *ifc = netif_get_state (netif);
if (ifc == NULL)
/* The user provided no interface */
return -1;
err = if_close (netif);
if (err)
return err;
return ifc->terminate (netif);
}
/* Args for _if_change_flags() */
struct if_change_flags_args
{
struct netif *netif;
uint16_t flags;
error_t err;
};
/*
* Implementation of if_change_flags(), called inside a thread-safe context
*/
static void
_if_change_flags (void *arg)
{
error_t err;
struct ifcommon *ifc;
uint16_t oldflags;
struct if_change_flags_args *args = arg;
ifc = netif_get_state (args->netif);
if (ifc == NULL)
{
/* The user provided no interface */
errno = EINVAL;
return;
}
oldflags = ifc->flags;
err = ifc->change_flags (args->netif, args->flags);
if (!err && ((oldflags ^ args->flags) & IFF_UP)) /* Bit is different ? */
err = ((oldflags & IFF_UP) ? if_close : if_open) (args->netif);
args->err = err;
return;
}
/*
* Change device flags.
*
* If IFF_UP changes, it opens/closes the device accordingly.
*/
error_t
if_change_flags (struct netif * netif, uint16_t flags)
{
error_t err;
/*
* Call _if_change_flags() inside the tcpip_thread and wait for it to finish.
*/
struct if_change_flags_args *args =
calloc (1, sizeof (struct if_change_flags_args));
args->netif = netif;
args->flags = flags;
err = tcpip_callback_wait(_if_change_flags, args);
if(!err)
/* Get the return value */
err = args->err;
free (args);
return err;
}