Remove _PROTOCOL macros for gcc 3.2 compatibility
[people/xl0/gpxe.git] / src / net / ethernet.c
1 /*
2  * Copyright (C) 2006 Michael Brown <mbrown@fensystems.co.uk>.
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License as
6  * published by the Free Software Foundation; either version 2 of the
7  * License, or any later version.
8  *
9  * This program is distributed in the hope that it will be useful, but
10  * WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
17  */
18
19 #include <stdint.h>
20 #include <string.h>
21 #include <byteswap.h>
22 #include <errno.h>
23 #include <assert.h>
24 #include <vsprintf.h>
25 #include <gpxe/if_arp.h>
26 #include <gpxe/if_ether.h>
27 #include <gpxe/netdevice.h>
28 #include <gpxe/pkbuff.h>
29 #include <gpxe/ethernet.h>
30
31 /** @file
32  *
33  * Ethernet protocol
34  *
35  */
36
37 /** Ethernet broadcast MAC address */
38 static uint8_t eth_broadcast[ETH_ALEN] = { 0xff, 0xff, 0xff, 0xff, 0xff, 0xff};
39
40 /**
41  * Transmit Ethernet packet
42  *
43  * @v pkb               Packet buffer
44  * @v netdev            Network device
45  * @v net_protocol      Network-layer protocol
46  * @v ll_dest           Link-layer destination address
47  *
48  * Prepends the Ethernet link-layer header and transmits the packet.
49  */
50 static int eth_tx ( struct pk_buff *pkb, struct net_device *netdev,
51                     struct net_protocol *net_protocol, const void *ll_dest ) {
52         struct ethhdr *ethhdr = pkb_push ( pkb, sizeof ( *ethhdr ) );
53
54         /* Build Ethernet header */
55         memcpy ( ethhdr->h_dest, ll_dest, ETH_ALEN );
56         memcpy ( ethhdr->h_source, netdev->ll_addr, ETH_ALEN );
57         ethhdr->h_protocol = net_protocol->net_proto;
58
59         /* Hand off to network device */
60         return netdev_tx ( netdev, pkb );
61 }
62
63 /**
64  * Process received Ethernet packet
65  *
66  * @v pkb       Packet buffer
67  * @v netdev    Network device
68  *
69  * Strips off the Ethernet link-layer header and passes up to the
70  * network-layer protocol.
71  */
72 static int eth_rx ( struct pk_buff *pkb, struct net_device *netdev ) {
73         struct ethhdr *ethhdr = pkb->data;
74
75         /* Sanity check */
76         if ( pkb_len ( pkb ) < sizeof ( *ethhdr ) ) {
77                 DBG ( "Ethernet packet too short (%d bytes)\n",
78                       pkb_len ( pkb ) );
79                 free_pkb ( pkb );
80                 return -EINVAL;
81         }
82
83         /* Strip off Ethernet header */
84         pkb_pull ( pkb, sizeof ( *ethhdr ) );
85
86         /* Hand off to network-layer protocol */
87         return net_rx ( pkb, netdev, ethhdr->h_protocol, ethhdr->h_source );
88 }
89
90 /**
91  * Transcribe Ethernet address
92  *
93  * @v ll_addr   Link-layer address
94  * @ret string  Link-layer address in human-readable format
95  */
96 static const char * eth_ntoa ( const void *ll_addr ) {
97         static char buf[18]; /* "00:00:00:00:00:00" */
98         const uint8_t *eth_addr = ll_addr;
99
100         sprintf ( buf, "%02x:%02x:%02x:%02x:%02x:%02x",
101                   eth_addr[0], eth_addr[1], eth_addr[2],
102                   eth_addr[3], eth_addr[4], eth_addr[5] );
103         return buf;
104 }
105
106 /** Ethernet protocol */
107 struct ll_protocol ethernet_protocol __ll_protocol = {
108         .name           = "Ethernet",
109         .ll_proto       = htons ( ARPHRD_ETHER ),
110         .ll_addr_len    = ETH_ALEN,
111         .ll_broadcast   = eth_broadcast,
112         .tx             = eth_tx,
113         .rx             = eth_rx,
114         .ntoa           = eth_ntoa,
115 };