1*58a2b000SEvgeniy Ivanov /* $NetBSD: ether.c,v 1.22 2009/01/12 11:32:45 tsutsui Exp $ */ 2*58a2b000SEvgeniy Ivanov 3*58a2b000SEvgeniy Ivanov /* 4*58a2b000SEvgeniy Ivanov * Copyright (c) 1992 Regents of the University of California. 5*58a2b000SEvgeniy Ivanov * All rights reserved. 6*58a2b000SEvgeniy Ivanov * 7*58a2b000SEvgeniy Ivanov * This software was developed by the Computer Systems Engineering group 8*58a2b000SEvgeniy Ivanov * at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and 9*58a2b000SEvgeniy Ivanov * contributed to Berkeley. 10*58a2b000SEvgeniy Ivanov * 11*58a2b000SEvgeniy Ivanov * Redistribution and use in source and binary forms, with or without 12*58a2b000SEvgeniy Ivanov * modification, are permitted provided that the following conditions 13*58a2b000SEvgeniy Ivanov * are met: 14*58a2b000SEvgeniy Ivanov * 1. Redistributions of source code must retain the above copyright 15*58a2b000SEvgeniy Ivanov * notice, this list of conditions and the following disclaimer. 16*58a2b000SEvgeniy Ivanov * 2. Redistributions in binary form must reproduce the above copyright 17*58a2b000SEvgeniy Ivanov * notice, this list of conditions and the following disclaimer in the 18*58a2b000SEvgeniy Ivanov * documentation and/or other materials provided with the distribution. 19*58a2b000SEvgeniy Ivanov * 3. All advertising materials mentioning features or use of this software 20*58a2b000SEvgeniy Ivanov * must display the following acknowledgement: 21*58a2b000SEvgeniy Ivanov * This product includes software developed by the University of 22*58a2b000SEvgeniy Ivanov * California, Lawrence Berkeley Laboratory and its contributors. 23*58a2b000SEvgeniy Ivanov * 4. Neither the name of the University nor the names of its contributors 24*58a2b000SEvgeniy Ivanov * may be used to endorse or promote products derived from this software 25*58a2b000SEvgeniy Ivanov * without specific prior written permission. 26*58a2b000SEvgeniy Ivanov * 27*58a2b000SEvgeniy Ivanov * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 28*58a2b000SEvgeniy Ivanov * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 29*58a2b000SEvgeniy Ivanov * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 30*58a2b000SEvgeniy Ivanov * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 31*58a2b000SEvgeniy Ivanov * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 32*58a2b000SEvgeniy Ivanov * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 33*58a2b000SEvgeniy Ivanov * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 34*58a2b000SEvgeniy Ivanov * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 35*58a2b000SEvgeniy Ivanov * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 36*58a2b000SEvgeniy Ivanov * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 37*58a2b000SEvgeniy Ivanov * SUCH DAMAGE. 38*58a2b000SEvgeniy Ivanov * 39*58a2b000SEvgeniy Ivanov * @(#) Header: net.c,v 1.9 93/08/06 19:32:15 leres Exp (LBL) 40*58a2b000SEvgeniy Ivanov */ 41*58a2b000SEvgeniy Ivanov 42*58a2b000SEvgeniy Ivanov #include <sys/param.h> 43*58a2b000SEvgeniy Ivanov #include <sys/socket.h> 44*58a2b000SEvgeniy Ivanov #ifdef _STANDALONE 45*58a2b000SEvgeniy Ivanov #include <lib/libkern/libkern.h> 46*58a2b000SEvgeniy Ivanov #else 47*58a2b000SEvgeniy Ivanov #include <string.h> 48*58a2b000SEvgeniy Ivanov #endif 49*58a2b000SEvgeniy Ivanov 50*58a2b000SEvgeniy Ivanov #include <net/if.h> 51*58a2b000SEvgeniy Ivanov #include <net/if_ether.h> 52*58a2b000SEvgeniy Ivanov 53*58a2b000SEvgeniy Ivanov #include <netinet/in.h> 54*58a2b000SEvgeniy Ivanov #include <netinet/in_systm.h> 55*58a2b000SEvgeniy Ivanov 56*58a2b000SEvgeniy Ivanov #include "stand.h" 57*58a2b000SEvgeniy Ivanov #include "net.h" 58*58a2b000SEvgeniy Ivanov 59*58a2b000SEvgeniy Ivanov /* Caller must leave room for ethernet header in front!! */ 60*58a2b000SEvgeniy Ivanov ssize_t 61*58a2b000SEvgeniy Ivanov sendether(struct iodesc *d, void *pkt, size_t len, u_char *dea, int etype) 62*58a2b000SEvgeniy Ivanov { 63*58a2b000SEvgeniy Ivanov ssize_t n; 64*58a2b000SEvgeniy Ivanov struct ether_header *eh; 65*58a2b000SEvgeniy Ivanov 66*58a2b000SEvgeniy Ivanov #ifdef ETHER_DEBUG 67*58a2b000SEvgeniy Ivanov if (debug) 68*58a2b000SEvgeniy Ivanov printf("sendether: called\n"); 69*58a2b000SEvgeniy Ivanov #endif 70*58a2b000SEvgeniy Ivanov 71*58a2b000SEvgeniy Ivanov eh = (struct ether_header *)pkt - 1; 72*58a2b000SEvgeniy Ivanov len += sizeof(*eh); 73*58a2b000SEvgeniy Ivanov 74*58a2b000SEvgeniy Ivanov MACPY(d->myea, eh->ether_shost); /* by byte */ 75*58a2b000SEvgeniy Ivanov MACPY(dea, eh->ether_dhost); /* by byte */ 76*58a2b000SEvgeniy Ivanov eh->ether_type = htons(etype); 77*58a2b000SEvgeniy Ivanov 78*58a2b000SEvgeniy Ivanov n = netif_put(d, eh, len); 79*58a2b000SEvgeniy Ivanov if (n == -1 || (size_t)n < sizeof(*eh)) 80*58a2b000SEvgeniy Ivanov return -1; 81*58a2b000SEvgeniy Ivanov 82*58a2b000SEvgeniy Ivanov n -= sizeof(*eh); 83*58a2b000SEvgeniy Ivanov return n; 84*58a2b000SEvgeniy Ivanov } 85*58a2b000SEvgeniy Ivanov 86*58a2b000SEvgeniy Ivanov /* 87*58a2b000SEvgeniy Ivanov * Get a packet of any Ethernet type, with our address or 88*58a2b000SEvgeniy Ivanov * the broadcast address. Save the Ether type in arg 5. 89*58a2b000SEvgeniy Ivanov * NOTE: Caller must leave room for the Ether header. 90*58a2b000SEvgeniy Ivanov */ 91*58a2b000SEvgeniy Ivanov ssize_t 92*58a2b000SEvgeniy Ivanov readether(struct iodesc *d, void *pkt, size_t len, saseconds_t tleft, 93*58a2b000SEvgeniy Ivanov u_int16_t *etype) 94*58a2b000SEvgeniy Ivanov { 95*58a2b000SEvgeniy Ivanov ssize_t n; 96*58a2b000SEvgeniy Ivanov struct ether_header *eh; 97*58a2b000SEvgeniy Ivanov 98*58a2b000SEvgeniy Ivanov #ifdef ETHER_DEBUG 99*58a2b000SEvgeniy Ivanov if (debug) 100*58a2b000SEvgeniy Ivanov printf("readether: called\n"); 101*58a2b000SEvgeniy Ivanov #endif 102*58a2b000SEvgeniy Ivanov 103*58a2b000SEvgeniy Ivanov eh = (struct ether_header *)pkt - 1; 104*58a2b000SEvgeniy Ivanov len += sizeof(*eh); 105*58a2b000SEvgeniy Ivanov 106*58a2b000SEvgeniy Ivanov n = netif_get(d, eh, len, tleft); 107*58a2b000SEvgeniy Ivanov if (n == -1 || (size_t)n < sizeof(*eh)) 108*58a2b000SEvgeniy Ivanov return -1; 109*58a2b000SEvgeniy Ivanov 110*58a2b000SEvgeniy Ivanov /* Validate Ethernet address. */ 111*58a2b000SEvgeniy Ivanov if (memcmp(d->myea, eh->ether_dhost, 6) != 0 && 112*58a2b000SEvgeniy Ivanov memcmp(bcea, eh->ether_dhost, 6) != 0) { 113*58a2b000SEvgeniy Ivanov #ifdef ETHER_DEBUG 114*58a2b000SEvgeniy Ivanov if (debug) 115*58a2b000SEvgeniy Ivanov printf("readether: not ours (ea=%s)\n", 116*58a2b000SEvgeniy Ivanov ether_sprintf(eh->ether_dhost)); 117*58a2b000SEvgeniy Ivanov #endif 118*58a2b000SEvgeniy Ivanov return -1; 119*58a2b000SEvgeniy Ivanov } 120*58a2b000SEvgeniy Ivanov *etype = ntohs(eh->ether_type); 121*58a2b000SEvgeniy Ivanov 122*58a2b000SEvgeniy Ivanov n -= sizeof(*eh); 123*58a2b000SEvgeniy Ivanov return n; 124*58a2b000SEvgeniy Ivanov } 125