Can now both send and receive packets. LL header format not yet
[people/xl0/gpxe.git] / src / net / infiniband.c
1 /*
2  * Copyright (C) 2007 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 <stdio.h>
21 #include <string.h>
22 #include <byteswap.h>
23 #include <errno.h>
24 #include <assert.h>
25 #include <gpxe/if_arp.h>
26 #include <gpxe/netdevice.h>
27 #include <gpxe/iobuf.h>
28 #include <gpxe/infiniband.h>
29
30 /** @file
31  *
32  * Infiniband protocol
33  *
34  */
35
36 /** Infiniband broadcast MAC address */
37 static uint8_t ib_broadcast[IB_ALEN] = { 0xff, };
38
39 /**
40  * Transmit Infiniband packet
41  *
42  * @v iobuf             I/O buffer
43  * @v netdev            Network device
44  * @v net_protocol      Network-layer protocol
45  * @v ll_dest           Link-layer destination address
46  *
47  * Prepends the Infiniband link-layer header and transmits the packet.
48  */
49 static int ib_tx ( struct io_buffer *iobuf, struct net_device *netdev,
50                    struct net_protocol *net_protocol, const void *ll_dest ) {
51         struct ibhdr *ibhdr = iob_push ( iobuf, sizeof ( *ibhdr ) );
52
53
54         /* Build Infiniband header */
55         memcpy ( ibhdr->peer, ll_dest, IB_ALEN );
56         ibhdr->proto = net_protocol->net_proto;
57         ibhdr->reserved = 0;
58
59         /* Hand off to network device */
60         return netdev_tx ( netdev, iobuf );
61 }
62
63 /**
64  * Process received Infiniband packet
65  *
66  * @v iobuf     I/O buffer
67  * @v netdev    Network device
68  *
69  * Strips off the Infiniband link-layer header and passes up to the
70  * network-layer protocol.
71  */
72 static int ib_rx ( struct io_buffer *iobuf, struct net_device *netdev ) {
73
74         struct {
75                 uint16_t proto;
76                 uint16_t reserved;
77         } * header = iobuf->data;
78
79         iob_pull ( iobuf, sizeof ( *header ) );
80         return net_rx ( iobuf, netdev, header->proto, NULL );
81
82
83
84         struct ibhdr *ibhdr = iobuf->data;
85
86         /* Sanity check */
87         if ( iob_len ( iobuf ) < sizeof ( *ibhdr ) ) {
88                 DBG ( "Infiniband packet too short (%d bytes)\n",
89                       iob_len ( iobuf ) );
90                 free_iob ( iobuf );
91                 return -EINVAL;
92         }
93
94         /* Strip off Infiniband header */
95         iob_pull ( iobuf, sizeof ( *ibhdr ) );
96
97         /* Hand off to network-layer protocol */
98         return net_rx ( iobuf, netdev, ibhdr->proto, ibhdr->peer );
99 }
100
101 /**
102  * Transcribe Infiniband address
103  *
104  * @v ll_addr   Link-layer address
105  * @ret string  Link-layer address in human-readable format
106  */
107 const char * ib_ntoa ( const void *ll_addr ) {
108         static char buf[61];
109         const uint8_t *ib_addr = ll_addr;
110         unsigned int i;
111         char *p = buf;
112
113         for ( i = 0 ; i < IB_ALEN ; i++ ) {
114                 p += sprintf ( p, ":%02x", ib_addr[i] );
115         }
116         return ( buf + 1 );
117 }
118
119 /** Infiniband protocol */
120 struct ll_protocol infiniband_protocol __ll_protocol = {
121         .name           = "Infiniband",
122         .ll_proto       = htons ( ARPHRD_INFINIBAND ),
123         .ll_addr_len    = IB_ALEN,
124         .ll_header_len  = IB_HLEN,
125         .ll_broadcast   = ib_broadcast,
126         .tx             = ib_tx,
127         .rx             = ib_rx,
128         .ntoa           = ib_ntoa,
129 };