blob: d73a957110efe802128ca8fac838e3cb9d353106 [file] [log] [blame]
/*
* Copyright (C) 2010 Michael Brown <mbrown@fensystems.co.uk>.
*
* This program 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 of the
* License, or any later version.
*
* This program 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 this program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
* 02110-1301, USA.
*
* You can also choose to distribute this program under the terms of
* the Unmodified Binary Distribution Licence (as given in the file
* COPYING.UBDL), provided that you have satisfied its requirements.
*/
FILE_LICENCE ( GPL2_OR_LATER_OR_UBDL );
#include <stdint.h>
#include <string.h>
#include <stdio.h>
#include <errno.h>
#include <byteswap.h>
#include <ipxe/features.h>
#include <ipxe/if_ether.h>
#include <ipxe/ethernet.h>
#include <ipxe/netdevice.h>
#include <ipxe/iobuf.h>
#include <ipxe/vlan.h>
/** @file
*
* Virtual LANs
*
*/
FEATURE ( FEATURE_PROTOCOL, "VLAN", DHCP_EB_FEATURE_VLAN, 1 );
struct net_protocol vlan_protocol __net_protocol;
/** VLAN device private data */
struct vlan_device {
/** Trunk network device */
struct net_device *trunk;
/** VLAN tag */
unsigned int tag;
/** Default priority */
unsigned int priority;
};
/** Automatic VLAN device link-layer address */
static uint8_t vlan_auto_ll_addr[ETH_ALEN];
/** Automatic VLAN tag */
static unsigned int vlan_auto_tag;
/**
* Open VLAN device
*
* @v netdev Network device
* @ret rc Return status code
*/
static int vlan_open ( struct net_device *netdev ) {
struct vlan_device *vlan = netdev->priv;
return netdev_open ( vlan->trunk );
}
/**
* Close VLAN device
*
* @v netdev Network device
*/
static void vlan_close ( struct net_device *netdev ) {
struct vlan_device *vlan = netdev->priv;
netdev_close ( vlan->trunk );
}
/**
* Transmit packet on VLAN device
*
* @v netdev Network device
* @v iobuf I/O buffer
* @ret rc Return status code
*/
static int vlan_transmit ( struct net_device *netdev,
struct io_buffer *iobuf ) {
struct vlan_device *vlan = netdev->priv;
struct net_device *trunk = vlan->trunk;
struct ll_protocol *ll_protocol;
struct vlan_header *vlanhdr;
uint8_t ll_dest_copy[ETH_ALEN];
uint8_t ll_source_copy[ETH_ALEN];
const void *ll_dest;
const void *ll_source;
uint16_t net_proto;
unsigned int flags;
int rc;
/* Strip link-layer header and preserve link-layer header fields */
ll_protocol = netdev->ll_protocol;
if ( ( rc = ll_protocol->pull ( netdev, iobuf, &ll_dest, &ll_source,
&net_proto, &flags ) ) != 0 ) {
DBGC ( netdev, "VLAN %s could not parse link-layer header: "
"%s\n", netdev->name, strerror ( rc ) );
return rc;
}
memcpy ( ll_dest_copy, ll_dest, ETH_ALEN );
memcpy ( ll_source_copy, ll_source, ETH_ALEN );
/* Construct VLAN header */
vlanhdr = iob_push ( iobuf, sizeof ( *vlanhdr ) );
vlanhdr->tci = htons ( VLAN_TCI ( vlan->tag, vlan->priority ) );
vlanhdr->net_proto = net_proto;
/* Reclaim I/O buffer from VLAN device's TX queue */
list_del ( &iobuf->list );
/* Transmit packet on trunk device */
if ( ( rc = net_tx ( iob_disown ( iobuf ), trunk, &vlan_protocol,
ll_dest_copy, ll_source_copy ) ) != 0 ) {
DBGC ( netdev, "VLAN %s could not transmit: %s\n",
netdev->name, strerror ( rc ) );
/* Cannot return an error status, since that would
* cause the I/O buffer to be double-freed.
*/
return 0;
}
return 0;
}
/**
* Poll VLAN device
*
* @v netdev Network device
*/
static void vlan_poll ( struct net_device *netdev ) {
struct vlan_device *vlan = netdev->priv;
/* Poll trunk device */
netdev_poll ( vlan->trunk );
}
/**
* Enable/disable interrupts on VLAN device
*
* @v netdev Network device
* @v enable Interrupts should be enabled
*/
static void vlan_irq ( struct net_device *netdev, int enable ) {
struct vlan_device *vlan = netdev->priv;
/* Enable/disable interrupts on trunk device. This is not at
* all robust, but there is no sensible course of action
* available.
*/
netdev_irq ( vlan->trunk, enable );
}
/** VLAN device operations */
static struct net_device_operations vlan_operations = {
.open = vlan_open,
.close = vlan_close,
.transmit = vlan_transmit,
.poll = vlan_poll,
.irq = vlan_irq,
};
/**
* Synchronise VLAN device
*
* @v netdev Network device
*/
static void vlan_sync ( struct net_device *netdev ) {
struct vlan_device *vlan = netdev->priv;
struct net_device *trunk = vlan->trunk;
/* Synchronise link status */
if ( netdev->link_rc != trunk->link_rc )
netdev_link_err ( netdev, trunk->link_rc );
/* Synchronise open/closed status */
if ( netdev_is_open ( trunk ) ) {
if ( ! netdev_is_open ( netdev ) )
netdev_open ( netdev );
} else {
if ( netdev_is_open ( netdev ) )
netdev_close ( netdev );
}
}
/**
* Identify VLAN device
*
* @v trunk Trunk network device
* @v tag VLAN tag
* @ret netdev VLAN device, if any
*/
struct net_device * vlan_find ( struct net_device *trunk, unsigned int tag ) {
struct net_device *netdev;
struct vlan_device *vlan;
for_each_netdev ( netdev ) {
if ( netdev->op != &vlan_operations )
continue;
vlan = netdev->priv;
if ( ( vlan->trunk == trunk ) && ( vlan->tag == tag ) )
return netdev;
}
return NULL;
}
/**
* Process incoming VLAN packet
*
* @v iobuf I/O buffer
* @v trunk Trunk network device
* @v ll_dest Link-layer destination address
* @v ll_source Link-layer source address
* @v flags Packet flags
* @ret rc Return status code
*/
static int vlan_rx ( struct io_buffer *iobuf, struct net_device *trunk,
const void *ll_dest, const void *ll_source,
unsigned int flags __unused ) {
struct vlan_header *vlanhdr = iobuf->data;
struct net_device *netdev;
struct ll_protocol *ll_protocol;
uint8_t ll_dest_copy[ETH_ALEN];
uint8_t ll_source_copy[ETH_ALEN];
uint16_t tag;
int rc;
/* Sanity check */
if ( iob_len ( iobuf ) < sizeof ( *vlanhdr ) ) {
DBGC ( trunk, "VLAN %s received underlength packet (%zd "
"bytes)\n", trunk->name, iob_len ( iobuf ) );
rc = -EINVAL;
goto err_sanity;
}
/* Identify VLAN device */
tag = VLAN_TAG ( ntohs ( vlanhdr->tci ) );
netdev = vlan_find ( trunk, tag );
if ( ! netdev ) {
DBGC2 ( trunk, "VLAN %s received packet for unknown VLAN "
"%d\n", trunk->name, tag );
rc = -EPIPE;
goto err_no_vlan;
}
/* Strip VLAN header and preserve original link-layer header fields */
iob_pull ( iobuf, sizeof ( *vlanhdr ) );
ll_protocol = trunk->ll_protocol;
memcpy ( ll_dest_copy, ll_dest, ETH_ALEN );
memcpy ( ll_source_copy, ll_source, ETH_ALEN );
/* Reconstruct link-layer header for VLAN device */
ll_protocol = netdev->ll_protocol;
if ( ( rc = ll_protocol->push ( netdev, iobuf, ll_dest_copy,
ll_source_copy,
vlanhdr->net_proto ) ) != 0 ) {
DBGC ( netdev, "VLAN %s could not reconstruct link-layer "
"header: %s\n", netdev->name, strerror ( rc ) );
goto err_ll_push;
}
/* Enqueue packet on VLAN device */
netdev_rx ( netdev, iob_disown ( iobuf ) );
return 0;
err_ll_push:
err_no_vlan:
err_sanity:
free_iob ( iobuf );
return rc;
}
/** VLAN protocol */
struct net_protocol vlan_protocol __net_protocol = {
.name = "VLAN",
.net_proto = htons ( ETH_P_8021Q ),
.rx = vlan_rx,
};
/**
* Get the VLAN tag control information
*
* @v netdev Network device
* @ret tci VLAN tag control information, or 0 if not a VLAN device
*/
unsigned int vlan_tci ( struct net_device *netdev ) {
struct vlan_device *vlan;
if ( netdev->op == &vlan_operations ) {
vlan = netdev->priv;
return ( VLAN_TCI ( vlan->tag, vlan->priority ) );
} else {
return 0;
}
}
/**
* Check if network device can be used as a VLAN trunk device
*
* @v trunk Trunk network device
* @ret is_ok Trunk network device is usable
*
* VLAN devices will be created as Ethernet devices. (We cannot
* simply clone the link layer of the trunk network device, because
* this link layer may expect the network device structure to contain
* some link-layer-private data.) The trunk network device must
* therefore have a link layer that is in some sense 'compatible' with
* Ethernet; specifically, it must have link-layer addresses that are
* the same length as Ethernet link-layer addresses.
*
* As an additional check, and primarily to assist with the sanity of
* the FCoE code, we refuse to allow nested VLANs.
*/
int vlan_can_be_trunk ( struct net_device *trunk ) {
return ( ( trunk->ll_protocol->ll_addr_len == ETH_ALEN ) &&
( trunk->op != &vlan_operations ) );
}
/**
* Create VLAN device
*
* @v trunk Trunk network device
* @v tag VLAN tag
* @v priority Default VLAN priority
* @ret rc Return status code
*/
int vlan_create ( struct net_device *trunk, unsigned int tag,
unsigned int priority ) {
struct net_device *netdev;
struct vlan_device *vlan;
int rc;
/* If VLAN already exists, just update the priority */
if ( ( netdev = vlan_find ( trunk, tag ) ) != NULL ) {
vlan = netdev->priv;
if ( priority != vlan->priority ) {
DBGC ( netdev, "VLAN %s priority changed from %d to "
"%d\n", netdev->name, vlan->priority, priority );
}
vlan->priority = priority;
return 0;
}
/* Sanity checks */
if ( ! vlan_can_be_trunk ( trunk ) ) {
DBGC ( trunk, "VLAN %s cannot create VLAN on non-trunk "
"device\n", trunk->name );
rc = -ENOTTY;
goto err_sanity;
}
if ( ! VLAN_TAG_IS_VALID ( tag ) ) {
DBGC ( trunk, "VLAN %s cannot create VLAN with invalid tag "
"%d\n", trunk->name, tag );
rc = -EINVAL;
goto err_sanity;
}
if ( ! VLAN_PRIORITY_IS_VALID ( priority ) ) {
DBGC ( trunk, "VLAN %s cannot create VLAN with invalid "
"priority %d\n", trunk->name, priority );
rc = -EINVAL;
goto err_sanity;
}
/* Allocate and initialise structure */
netdev = alloc_etherdev ( sizeof ( *vlan ) );
if ( ! netdev ) {
rc = -ENOMEM;
goto err_alloc_etherdev;
}
netdev_init ( netdev, &vlan_operations );
netdev->dev = trunk->dev;
memcpy ( netdev->hw_addr, trunk->ll_addr, ETH_ALEN );
vlan = netdev->priv;
vlan->trunk = netdev_get ( trunk );
vlan->tag = tag;
vlan->priority = priority;
/* Construct VLAN device name */
snprintf ( netdev->name, sizeof ( netdev->name ), "%s-%d",
trunk->name, vlan->tag );
/* Mark device as not supporting interrupts, if applicable */
if ( ! netdev_irq_supported ( trunk ) )
netdev->state |= NETDEV_IRQ_UNSUPPORTED;
/* Register VLAN device */
if ( ( rc = register_netdev ( netdev ) ) != 0 ) {
DBGC ( netdev, "VLAN %s could not register: %s\n",
netdev->name, strerror ( rc ) );
goto err_register;
}
/* Synchronise with trunk device */
vlan_sync ( netdev );
DBGC ( netdev, "VLAN %s created with tag %d and priority %d\n",
netdev->name, vlan->tag, vlan->priority );
return 0;
unregister_netdev ( netdev );
err_register:
netdev_nullify ( netdev );
netdev_put ( netdev );
netdev_put ( trunk );
err_alloc_etherdev:
err_sanity:
return rc;
}
/**
* Destroy VLAN device
*
* @v netdev Network device
* @ret rc Return status code
*/
int vlan_destroy ( struct net_device *netdev ) {
struct vlan_device *vlan = netdev->priv;
struct net_device *trunk;
/* Sanity check */
if ( netdev->op != &vlan_operations ) {
DBGC ( netdev, "VLAN %s cannot destroy non-VLAN device\n",
netdev->name );
return -ENOTTY;
}
DBGC ( netdev, "VLAN %s destroyed\n", netdev->name );
/* Remove VLAN device */
unregister_netdev ( netdev );
trunk = vlan->trunk;
netdev_nullify ( netdev );
netdev_put ( netdev );
netdev_put ( trunk );
return 0;
}
/**
* Configure automatic VLAN device
*
* @v ll_addr Link-layer address
* @v tag VLAN tag
*/
void vlan_auto ( const void *ll_addr, unsigned int tag ) {
/* Record link-layer address and VLAN tag */
memcpy ( vlan_auto_ll_addr, ll_addr, ETH_ALEN );
vlan_auto_tag = tag;
}
/**
* Create automatic VLAN device
*
* @v trunk Trunk network device
* @ret rc Return status code
*/
static int vlan_probe ( struct net_device *trunk ) {
int rc;
/* Do nothing unless an automatic VLAN exists */
if ( ! vlan_auto_tag )
return 0;
/* Ignore non-trunk devices */
if ( ! vlan_can_be_trunk ( trunk ) )
return 0;
/* Ignore non-matching link-layer addresses */
if ( memcmp ( trunk->ll_addr, vlan_auto_ll_addr, ETH_ALEN ) != 0 )
return 0;
/* Create automatic VLAN device */
if ( ( rc = vlan_create ( trunk, vlan_auto_tag, 0 ) ) != 0 )
return rc;
return 0;
}
/**
* Handle trunk network device link state change
*
* @v trunk Trunk network device
*/
static void vlan_notify ( struct net_device *trunk ) {
struct net_device *netdev;
struct vlan_device *vlan;
for_each_netdev ( netdev ) {
if ( netdev->op != &vlan_operations )
continue;
vlan = netdev->priv;
if ( vlan->trunk == trunk )
vlan_sync ( netdev );
}
}
/**
* Destroy first VLAN device for a given trunk
*
* @v trunk Trunk network device
* @ret found A VLAN device was found
*/
static int vlan_remove_first ( struct net_device *trunk ) {
struct net_device *netdev;
struct vlan_device *vlan;
for_each_netdev ( netdev ) {
if ( netdev->op != &vlan_operations )
continue;
vlan = netdev->priv;
if ( vlan->trunk == trunk ) {
vlan_destroy ( netdev );
return 1;
}
}
return 0;
}
/**
* Destroy all VLAN devices for a given trunk
*
* @v trunk Trunk network device
*/
static void vlan_remove ( struct net_device *trunk ) {
/* Remove all VLAN devices attached to this trunk, safe
* against arbitrary net device removal.
*/
while ( vlan_remove_first ( trunk ) ) {}
}
/** VLAN driver */
struct net_driver vlan_driver __net_driver = {
.name = "VLAN",
.probe = vlan_probe,
.notify = vlan_notify,
.remove = vlan_remove,
};
/**
* Add VLAN tag-stripped packet to receive queue
*
* @v netdev Network device
* @v tag VLAN tag, or zero
* @v iobuf I/O buffer
*/
void vlan_netdev_rx ( struct net_device *netdev, unsigned int tag,
struct io_buffer *iobuf ) {
struct net_device *vlan;
/* Identify VLAN device, if applicable */
if ( tag ) {
if ( ( vlan = vlan_find ( netdev, tag ) ) == NULL ) {
netdev_rx_err ( netdev, iobuf, -ENODEV );
return;
}
netdev = vlan;
}
/* Hand off to network device */
netdev_rx ( netdev, iobuf );
}
/**
* Discard received VLAN tag-stripped packet
*
* @v netdev Network device
* @v tag VLAN tag, or zero
* @v iobuf I/O buffer, or NULL
* @v rc Packet status code
*/
void vlan_netdev_rx_err ( struct net_device *netdev, unsigned int tag,
struct io_buffer *iobuf, int rc ) {
struct net_device *vlan;
/* Identify VLAN device, if applicable */
if ( tag && ( ( vlan = vlan_find ( netdev, tag ) ) != NULL ) )
netdev = vlan;
/* Hand off to network device */
netdev_rx_err ( netdev, iobuf, rc );
}