Annotation of embedaddon/libnet/src/libnet_link_dlpi.c, revision 1.1.1.2

1.1       misho       1: /*
                      2:  *  $Id: libnet_link_dlpi.c,v 1.8 2004/01/28 19:45:00 mike Exp $
                      3:  *
                      4:  *  libnet
                      5:  *  libnet_dlpi.c - dlpi routines
                      6:  *
                      7:  *  Copyright (c) 1998 - 2004 Mike D. Schiffman <mike@infonexus.com>
                      8:  *  All rights reserved.
                      9:  *
                     10:  * Copyright (c) 1993, 1994, 1995, 1996, 1997
                     11:  *     The Regents of the University of California.  All rights reserved.
                     12:  *
                     13:  * Redistribution and use in source and binary forms, with or without
                     14:  * modification, are permitted provided that: (1) source code distributions
                     15:  * retain the above copyright notice and this paragraph in its entirety, (2)
                     16:  * distributions including binary code include the above copyright notice and
                     17:  * this paragraph in its entirety in the documentation or other materials
                     18:  * provided with the distribution, and (3) all advertising materials mentioning
                     19:  * features or use of this software display the following acknowledgement:
                     20:  * ``This product includes software developed by the University of California,
                     21:  * Lawrence Berkeley Laboratory and its contributors.'' Neither the name of
                     22:  * the University nor the names of its contributors may be used to endorse
                     23:  * or promote products derived from this software without specific prior
                     24:  * written permission.
                     25:  * THIS SOFTWARE IS PROVIDED ``AS IS'' AND WITHOUT ANY EXPRESS OR IMPLIED
                     26:  * WARRANTIES, INCLUDING, WITHOUT LIMITATION, THE IMPLIED WARRANTIES OF
                     27:  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.
                     28:  *
                     29:  * This code contributed by Atanu Ghosh (atanu@cs.ucl.ac.uk),
                     30:  * University College London.
                     31:  */
                     32: 
                     33: 
                     34: #if (HAVE_CONFIG_H)
                     35: #include "../include/config.h"
                     36: #endif
                     37: #include <sys/types.h>
                     38: #include <sys/time.h>
                     39: #ifdef HAVE_SYS_BUFMOD_H
                     40: #include <sys/bufmod.h>
                     41: #endif
                     42: #include <sys/dlpi.h>
                     43: #ifdef HAVE_HPUX9
                     44: #include <sys/socket.h>
                     45: #endif
                     46: #ifdef DL_HP_PPA_ACK_OBS
                     47: #include <sys/stat.h>
                     48: #endif
                     49: #include <sys/stream.h>
                     50: #if defined(HAVE_SOLARIS) && defined(HAVE_SYS_BUFMOD_H)
                     51: #include <sys/systeminfo.h>
                     52: #endif
                     53: 
                     54: #ifdef HAVE_SYS_DLPI_EXT_H
                     55: #include <sys/dlpi_ext.h>
                     56: #endif
                     57: 
                     58: #ifdef HAVE_HPUX9
                     59: #include <net/if.h>
                     60: #endif
                     61: 
                     62: #include <ctype.h>
                     63: #ifdef HAVE_HPUX9
                     64: #include <nlist.h>
                     65: #include <dlpi_ext.h>
                     66: #endif
                     67: #include <errno.h>
                     68: #include <fcntl.h>
                     69: #include <memory.h>
                     70: #include <stdio.h>
                     71: #include <stdlib.h>
                     72: #include <string.h>
                     73: #include <stropts.h>
                     74: #include <unistd.h>
                     75: 
                     76: #include "../include/libnet.h"
                     77: #include "../include/bpf.h"
                     78: 
                     79: #include "../include/gnuc.h"
                     80: #ifdef HAVE_OS_PROTO_H
                     81: #include "../include/os-proto.h"
                     82: #endif
                     83: 
                     84: #ifndef DLPI_DEV_PREFIX
                     85: #define DLPI_DEV_PREFIX "/dev"
                     86: #endif
                     87: 
                     88: #define        MAXDLBUF 8192
                     89: 
                     90: /* Forwards */
                     91: static int dlattachreq(int, bpf_u_int32, int8_t *);
                     92: static int dlbindack(int, int8_t *, int8_t *);
                     93: static int dlbindreq(int, bpf_u_int32, int8_t *);
                     94: static int dlinfoack(int, int8_t *, int8_t *);
                     95: static int dlinforeq(int, int8_t *);
                     96: static int dlokack(int, const int8_t *, int8_t *, int8_t *);
                     97: static int recv_ack(int, int, const int8_t *, int8_t *, int8_t *);
                     98: static int send_request(int, int8_t *, int, int8_t *, int8_t *, int);
                     99: #ifdef HAVE_SYS_BUFMOD_H
                    100: static int strioctl(int, int, int, int8_t *);
                    101: #endif
                    102: #ifdef HAVE_HPUX9
1.1.1.2 ! misho     103: static int dlpi_kread(int, off_t, void *, uint, int8_t *);
1.1       misho     104: #endif
                    105: #ifdef HAVE_DEV_DLPI
                    106: static int get_dlpi_ppa(int, const int8_t *, int, int8_t *);
                    107: #endif
                    108: 
                    109: /* XXX Needed by HP-UX (at least) */
                    110: static bpf_u_int32 ctlbuf[MAXDLBUF];
                    111: 
                    112: 
                    113: int
                    114: libnet_open_link(libnet_t *l)
                    115: {
                    116:     register int8_t *cp;
                    117:     int8_t *eos;
                    118:     register int ppa;
                    119:     register dl_info_ack_t *infop;
                    120:     bpf_u_int32 buf[MAXDLBUF];
                    121:     int8_t dname[100];
                    122: #ifndef HAVE_DEV_DLPI
                    123:     int8_t dname2[100];
                    124: #endif
                    125: 
                    126:     if (l == NULL)
                    127:     { 
                    128:         return (-1);
                    129:     } 
                    130: 
                    131:     /*
                    132:      *  Determine device and ppa
                    133:      */
                    134:     cp = strpbrk(l->device, "0123456789");
                    135:     if (cp == NULL)
                    136:     {
                    137:         snprintf(l->err_buf, LIBNET_ERRBUF_SIZE,
                    138:                 "%s(): %s is missing unit number\n", __func__, l->device);
                    139:         goto bad;
                    140:     }
                    141:     ppa = strtol(cp, &eos, 10);
                    142:     if (*eos != '\0')
                    143:     {
                    144:         snprintf(l->err_buf, LIBNET_ERRBUF_SIZE,
                    145:                 "%s(): %s bad unit number\n", __func__, l->device);
                    146:         goto bad;
                    147:     }
                    148: 
                    149:     if (*(l->device) == '/')
                    150:     {
                    151:         memset(&dname, 0, sizeof(dname));
                    152:         strncpy(dname, l->device, sizeof(dname) - 1);
                    153:         dname[sizeof(dname) - 1] = '\0';
                    154:     }
                    155:     else
                    156:     {
                    157:         sprintf(dname, "%s/%s", DLPI_DEV_PREFIX, l->device);
                    158:     }
                    159: #ifdef HAVE_DEV_DLPI
                    160:     /*
                    161:      *  Map network device to /dev/dlpi unit
                    162:      */
                    163:     cp = "/dev/dlpi";
                    164: 
                    165:     l->fd = open(cp, O_RDWR);
                    166:     if (l->fd == -1)
                    167:     {
                    168:         snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, "%s(): open(): %s: %s\n",
                    169:                 __func__, cp, strerror(errno));
                    170:         goto bad;
                    171:     }
                    172: 
                    173:     /*
                    174:      *  Map network interface to /dev/dlpi unit
                    175:      */
                    176:     ppa = get_dlpi_ppa(l->fd, dname, ppa, l->err_buf);
                    177:     if (ppa < 0)
                    178:     {
                    179:         goto bad;
                    180:     }
                    181: #else
                    182:     /*
                    183:      *  Try device without unit number
                    184:      */
                    185:     strcpy(dname2, dname);
                    186:     cp = strchr(dname, *cp);
                    187:     *cp = '\0';
                    188: 
                    189:     l->fd = open(dname, O_RDWR);
                    190:     if (l->fd == -1)
                    191:     {
                    192:         if (errno != ENOENT)
                    193:         {
                    194:             snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, "%s(): open(): %s: %s\n",
                    195:                     __func__, dname, strerror(errno));
                    196:             goto bad;
                    197:         }
                    198: 
                    199:         /*
                    200:          *  Try again with unit number
                    201:          */
                    202:         l->fd = open(dname2, O_RDWR);
                    203:         if (l->fd == -1)
                    204:         {
                    205:             snprintf(l->err_buf, LIBNET_ERRBUF_SIZE, "%s(): open(): %s: %s\n",
                    206:                     __func__, dname2, strerror(errno));
                    207:             goto bad;
                    208:         }
                    209: 
                    210:         cp = dname2;
                    211:         while (*cp && !isdigit((int)*cp))
                    212:         {
                    213:             cp++;
                    214:         }
                    215:         if (*cp)
                    216:         {
                    217:             ppa = atoi(cp);
                    218:         }
                    219:         else
                    220:         /*
                    221:          *  XXX Assume unit zero
                    222:          */
                    223:         ppa = 0;
                    224:     }
                    225: #endif
                    226:     /*
                    227:      *  Attach if "style 2" provider
                    228:      */
                    229:     if (dlinforeq(l->fd, l->err_buf) < 0 ||
                    230:             dlinfoack(l->fd, (int8_t *)buf, l->err_buf) < 0)
                    231:     {
                    232:         goto bad;
                    233:     }
                    234:     infop = &((union DL_primitives *)buf)->info_ack;
                    235:     if (infop->dl_provider_style == DL_STYLE2 &&
                    236:             (dlattachreq(l->fd, ppa, l->err_buf)
                    237:             < 0 || dlokack(l->fd, "attach", (int8_t *)buf, l->err_buf) < 0))
                    238:     {
                    239:         goto bad;
                    240:     }
                    241: 
                    242:     /*
                    243:      *  Bind HP-UX 9 and HP-UX 10.20
                    244:      */
                    245: #if defined(HAVE_HPUX9) || defined(HAVE_HPUX10_20) || defined(HAVE_HPUX11) || defined(HAVE_SOLARIS)
                    246:     if (dlbindreq(l->fd, 0, l->err_buf) < 0 ||
                    247:             dlbindack(l->fd, (int8_t *)buf, l->err_buf) < 0)
                    248:     {
                    249:         goto bad;
                    250:     }
                    251: #endif
                    252: 
                    253:     /*
                    254:      *  Determine link type
                    255:      */
                    256:     if (dlinforeq(l->fd, l->err_buf) < 0 ||
                    257:             dlinfoack(l->fd, (int8_t *)buf, l->err_buf) < 0)
                    258:     {
                    259:         goto bad;
                    260:     }
                    261: 
                    262:     infop = &((union DL_primitives *)buf)->info_ack;
                    263:     switch (infop->dl_mac_type)
                    264:     {
                    265:         case DL_CSMACD:
                    266:         case DL_ETHER:
                    267:             l->link_type    = DLT_EN10MB;
                    268:             l->link_offset  = 0xe;
                    269:             break;
                    270:         case DL_FDDI:
                    271:             l->link_type    = DLT_FDDI;
                    272:             l->link_offset  = 0x15;
                    273:             break;
                    274:         case DL_TPR:
                    275:             l->link_type    = DLT_PRONET;
                    276:             l->link_offset  = 0x16;
                    277:             break;
                    278:         default:
                    279:             sprintf(l->err_buf, "%s(): unknown mac type 0x%lu\n", __func__,
1.1.1.2 ! misho     280:                     (uint32_t) infop->dl_mac_type);
1.1       misho     281:             goto bad;
                    282:     }
                    283: 
                    284: #ifdef DLIOCRAW
                    285:     /*
                    286:      *  This is a non standard SunOS hack to get the ethernet header.
                    287:      */
                    288:     if (strioctl(l->fd, DLIOCRAW, 0, NULL) < 0)
                    289:     {
                    290:         sprintf(l->err_buf, "%s(): DLIOCRAW: %s\n", __func__, strerror(errno));
                    291:         goto bad;
                    292:     }
                    293: #endif
                    294: 
                    295:     return (1);
                    296: bad:
                    297:     if (l->fd > 0)
                    298:     {
                    299:         close(l->fd);      /* this can fail ok */
                    300:     }
                    301:     return (-1);
                    302: }
                    303: 
                    304: 
                    305: static int
                    306: send_request(int fd, int8_t *ptr, int len, int8_t *what, int8_t *ebuf,
                    307: int flags)
                    308: {
                    309:     struct strbuf ctl;
                    310: 
                    311:     ctl.maxlen = 0;
                    312:     ctl.len = len;
                    313:     ctl.buf = ptr;
                    314: 
                    315:     if (putmsg(fd, &ctl, (struct strbuf *) NULL, flags) < 0)
                    316:     {
                    317:         sprintf(ebuf, "%s(): putmsg \"%s\": %s\n", __func__, what,
                    318:                 strerror(errno));
                    319:         return (-1);
                    320:     }
                    321:     return (0);
                    322: }
                    323: 
                    324: static int
                    325: recv_ack(int fd, int size, const int8_t *what, int8_t *bufp, int8_t *ebuf)
                    326: {
                    327:     union DL_primitives *dlp;
                    328:     struct strbuf ctl;
                    329:     int flags;
                    330: 
                    331:     ctl.maxlen = MAXDLBUF;
                    332:     ctl.len = 0;
                    333:     ctl.buf = bufp;
                    334: 
                    335:     flags = 0;
                    336:     if (getmsg(fd, &ctl, (struct strbuf*)NULL, &flags) < 0)
                    337:     {
                    338:         sprintf(ebuf, "%s(): %s getmsg: %s\n", __func__, what, strerror(errno));
                    339:         return (-1);
                    340:     }
                    341: 
                    342:     dlp = (union DL_primitives *)ctl.buf;
                    343:     switch (dlp->dl_primitive)
                    344:     {
                    345:         case DL_INFO_ACK:
                    346:         case DL_PHYS_ADDR_ACK:
                    347:         case DL_BIND_ACK:
                    348:         case DL_OK_ACK:
                    349: #ifdef DL_HP_PPA_ACK
                    350:         case DL_HP_PPA_ACK:
                    351: #endif
                    352:         /*
                    353:          *  These are OK
                    354:          */
                    355:         break;
                    356: 
                    357:         case DL_ERROR_ACK:
                    358:             switch (dlp->error_ack.dl_errno)
                    359:             {
                    360:                 case DL_BADPPA:
                    361:                     sprintf(ebuf, "recv_ack: %s bad ppa (device unit)", what);
                    362:                     break;
                    363:                 case DL_SYSERR:
                    364:                     sprintf(ebuf, "recv_ack: %s: %s",
                    365:                         what, strerror(dlp->error_ack.dl_unix_errno));
                    366:                     break;
                    367:                 case DL_UNSUPPORTED:
                    368:                     sprintf(ebuf,
                    369:                         "recv_ack: %s: Service not supplied by provider", what);
                    370:                     break;
                    371:                 default:
                    372:                     sprintf(ebuf, "recv_ack: %s error 0x%x", what,
                    373:                         (bpf_u_int32)dlp->error_ack.dl_errno);
                    374:                     break;
                    375:             }
                    376:             return (-1);
                    377: 
                    378:         default:
                    379:             sprintf(ebuf, "recv_ack: %s unexpected primitive ack 0x%x ",
                    380:                 what, (bpf_u_int32)dlp->dl_primitive);
                    381:             return (-1);
                    382:     }
                    383: 
                    384:     if (ctl.len < size)
                    385:     {
                    386:         sprintf(ebuf, "recv_ack: %s ack too small (%d < %d)",
                    387:             what, ctl.len, size);
                    388:         return (-1);
                    389:     }
                    390:     return (ctl.len);
                    391: }
                    392: 
                    393: static int
                    394: dlattachreq(int fd, bpf_u_int32 ppa, int8_t *ebuf)
                    395: {
                    396:     dl_attach_req_t req;
                    397: 
                    398:     req.dl_primitive = DL_ATTACH_REQ;
                    399:     req.dl_ppa       = ppa;
                    400: 
                    401:     return (send_request(fd, (int8_t *)&req, sizeof(req), "attach", ebuf, 0));
                    402: }
                    403: 
                    404: static int
                    405: dlbindreq(int fd, bpf_u_int32 sap, int8_t *ebuf)
                    406: {
                    407: 
                    408:     dl_bind_req_t      req;
                    409: 
                    410:     memset((int8_t *)&req, 0, sizeof(req));
                    411:     req.dl_primitive = DL_BIND_REQ;
                    412: #ifdef DL_HP_RAWDLS
                    413:     req.dl_max_conind = 1;  /* XXX magic number */
                    414:     /*
                    415:      *  22 is INSAP as per the HP-UX DLPI Programmer's Guide
                    416:      */
                    417:     req.dl_sap = 22;
                    418:     req.dl_service_mode = DL_HP_RAWDLS;
                    419: #else
                    420:     req.dl_sap = sap;
                    421: #ifdef DL_CLDLS
                    422:     req.dl_service_mode = DL_CLDLS;
                    423: #endif
                    424: #endif
                    425:     return (send_request(fd, (int8_t *)&req, sizeof(req), "bind", ebuf, 0));
                    426: }
                    427: 
                    428: static int
                    429: dlbindack(int fd, int8_t *bufp, int8_t *ebuf)
                    430: {
                    431:     return (recv_ack(fd, DL_BIND_ACK_SIZE, "bind", bufp, ebuf));
                    432: }
                    433: 
                    434: static int
                    435: dlokack(int fd, const int8_t *what, int8_t *bufp, int8_t *ebuf)
                    436: {
                    437:     return (recv_ack(fd, DL_OK_ACK_SIZE, what, bufp, ebuf));
                    438: }
                    439: 
                    440: static int
                    441: dlinforeq(int fd, int8_t *ebuf)
                    442: {
                    443:     dl_info_req_t req;
                    444: 
                    445:     req.dl_primitive = DL_INFO_REQ;
                    446: 
                    447:     return (send_request(fd, (int8_t *)&req, sizeof(req), "info", ebuf, 
                    448:             RS_HIPRI));
                    449: }
                    450: 
                    451: static int
                    452: dlinfoack(int fd, int8_t *bufp, int8_t *ebuf)
                    453: {
                    454:     return (recv_ack(fd, DL_INFO_ACK_SIZE, "info", bufp, ebuf));
                    455: }
                    456: 
                    457: 
                    458: #ifdef HAVE_SYS_BUFMOD_H
                    459: static int
                    460: strioctl(int fd, int cmd, int len, int8_t *dp)
                    461: {
                    462:     struct strioctl str;
                    463:     int rc;
                    464: 
                    465:     str.ic_cmd    = cmd;
                    466:     str.ic_timout = -1;
                    467:     str.ic_len    = len;
                    468:     str.ic_dp     = dp;
                    469:     
                    470:     rc = ioctl(fd, I_STR, &str);
                    471:     if (rc < 0)
                    472:     {
                    473:         return (rc);
                    474:     }
                    475:     else
                    476:     {
                    477:         return (str.ic_len);
                    478:     }
                    479: }
                    480: #endif
                    481: 
                    482: 
                    483: #if (defined(DL_HP_PPA_ACK_OBS) && !defined(HAVE_HPUX11))
                    484: /*
                    485:  * Under HP-UX 10, we can ask for the ppa
                    486:  */
                    487: static int
                    488: get_dlpi_ppa(register int fd, register const int8_t *device, register int unit,
                    489: register int8_t *ebuf)
                    490: {
                    491:     register dl_hp_ppa_ack_t *ap;
                    492:     register dl_hp_ppa_info_t *ip;
                    493:     register int i;
1.1.1.2 ! misho     494:     register uint32_t majdev;
1.1       misho     495:     dl_hp_ppa_req_t    req;
                    496:     struct stat statbuf;
                    497:     bpf_u_int32 buf[MAXDLBUF];
                    498: 
                    499:     if (stat(device, &statbuf) < 0)
                    500:     {
                    501:         sprintf(ebuf, "stat: %s: %s", device, strerror(errno));
                    502:         return (-1);
                    503:     }
                    504:     majdev = major(statbuf.st_rdev);
                    505: 
                    506:     memset((int8_t *)&req, 0, sizeof(req));
                    507:     req.dl_primitive = DL_HP_PPA_REQ;
                    508: 
                    509:     memset((int8_t *)buf, 0, sizeof(buf));
                    510:     if (send_request(fd, (int8_t *)&req, sizeof(req), "hpppa", ebuf, 0) < 0 ||
                    511:         recv_ack(fd, DL_HP_PPA_ACK_SIZE, "hpppa", (int8_t *)buf, ebuf) < 0)
                    512:     {
                    513:         return (-1);
                    514:     }
                    515: 
                    516:     ap = (dl_hp_ppa_ack_t *)buf;
1.1.1.2 ! misho     517:     ip = (dl_hp_ppa_info_t *)((uint8_t *)ap + ap->dl_offset);
1.1       misho     518: 
                    519:     for (i = 0; i < ap->dl_count; i++)
                    520:     {
                    521:         if (ip->dl_mjr_num == majdev && ip->dl_instance_num == unit)
                    522:         break;
                    523: 
1.1.1.2 ! misho     524:         ip = (dl_hp_ppa_info_t *)((uint8_t *)ip + ip->dl_next_offset);
1.1       misho     525:     }
                    526: 
                    527:     if (i == ap->dl_count)
                    528:     {
                    529:         sprintf(ebuf, "can't find PPA for %s", device);
                    530:         return (-1);
                    531:     }
                    532: 
                    533:     if (ip->dl_hdw_state == HDW_DEAD)
                    534:     {
                    535:         sprintf(ebuf, "%s: hardware state: DOWN\n", device);
                    536:         return (-1);
                    537:     }
                    538:     return ((int)ip->dl_ppa);
                    539: }
                    540: #endif
                    541: 
                    542: #ifdef HAVE_HPUX9
                    543: /*
                    544:  * Under HP-UX 9, there is no good way to determine the ppa.
                    545:  * So punt and read it from /dev/kmem.
                    546:  */
                    547: static struct nlist nl[] =
                    548: {
                    549: #define NL_IFNET 0
                    550:     { "ifnet" },
                    551:     { "" }
                    552: };
                    553: 
                    554: static int8_t path_vmunix[] = "/hp-ux";
                    555: 
                    556: /*
                    557:  *  Determine ppa number that specifies ifname
                    558:  */
                    559: static int
                    560: get_dlpi_ppa(register int fd, register const int8_t *ifname, register int unit,
                    561:     register int8_t *ebuf)
                    562: {
                    563:     register const int8_t *cp;
                    564:     register int kd;
                    565:     void *addr;
                    566:     struct ifnet ifnet;
                    567:     int8_t if_name[sizeof(ifnet.if_name)], tifname[32];
                    568: 
                    569:     cp = strrchr(ifname, '/');
                    570:     if (cp != NULL)
                    571:     {
                    572:         ifname = cp + 1;
                    573:     }
                    574:     if (nlist(path_vmunix, &nl) < 0)
                    575:     {
                    576:         sprintf(ebuf, "nlist %s failed", path_vmunix);
                    577:         return (-1);
                    578:     }
                    579: 
                    580:     if (nl[NL_IFNET].n_value == 0)
                    581:     {
                    582:         sprintf(ebuf, "could't find %s kernel symbol", nl[NL_IFNET].n_name);
                    583:         return (-1);
                    584:     }
                    585: 
                    586:     kd = open("/dev/kmem", O_RDONLY);
                    587:     if (kd < 0)
                    588:     {
                    589:         sprintf(ebuf, "kmem open: %s", strerror(errno));
                    590:         return (-1);
                    591:     }
                    592: 
                    593:     if (dlpi_kread(kd, nl[NL_IFNET].n_value, &addr, sizeof(addr), ebuf) < 0)
                    594:     {
                    595:         close(kd);
                    596:         return (-1);
                    597:     }
                    598:     for (; addr != NULL; addr = ifnet.if_next)
                    599:     {
                    600:         if (dlpi_kread(kd, (off_t)addr, &ifnet, sizeof(ifnet), ebuf) < 0 ||
                    601:             dlpi_kread(kd, (off_t)ifnet.if_name,
                    602:             if_name, sizeof(if_name), ebuf) < 0)
                    603:             {
                    604:                 close(kd);
                    605:                 return (-1);
                    606:             }
                    607:             sprintf(tifname, "%.*s%d",
                    608:                 (int)sizeof(if_name), if_name, ifnet.if_unit);
                    609:             if (strcmp(tifname, ifname) == 0)
                    610:             {
                    611:                 return (ifnet.if_index);
                    612:             }
                    613:     }
                    614: 
                    615:     sprintf(ebuf, "Can't find %s", ifname);
                    616:     return (-1);
                    617: }
                    618: 
                    619: static int
                    620: dlpi_kread(register int fd, register off_t addr, register void *buf,
1.1.1.2 ! misho     621: register uint len, register int8_t *ebuf)
1.1       misho     622: {
                    623:     register int cc;
                    624: 
                    625:     if (lseek(fd, addr, SEEK_SET) < 0)
                    626:     {
                    627:         sprintf(ebuf, "lseek: %s", strerror(errno));
                    628:         return (-1);
                    629:     }
                    630:     cc = read(fd, buf, len);
                    631:     if (cc < 0)
                    632:     {
                    633:         sprintf(ebuf, "read: %s", strerror(errno));
                    634:         return (-1);
                    635:     }
                    636:     else if (cc != len)
                    637:     {
                    638:         sprintf(ebuf, "int16_t read (%d != %d)", cc, len);
                    639:         return (-1);
                    640:     }
                    641:     return (cc);
                    642: }
                    643: #endif
                    644: 
                    645: #define ETHERADDRL 6
                    646: struct  EnetHeaderInfo
                    647: {
                    648:     struct libnet_ether_addr   DestEtherAddr;
1.1.1.2 ! misho     649:     uint16_t EtherFrameType;
1.1       misho     650: };
                    651: 
                    652: 
                    653: int
                    654: libnet_close_link(libnet_t *l)
                    655: {
                    656:     if (close(l->fd) == 0)
                    657:     {
                    658:         return (1);
                    659:     }
                    660:     else
                    661:     {
                    662:         return (-1);
                    663:     }
                    664: }
                    665: 
                    666: #ifdef HAVE_HPUX11
                    667: int 
1.1.1.2 ! misho     668: libnet_write_link(libnet_t *l, const uint8_t *packet, uint32_t size)   
1.1       misho     669: {                                                             
                    670:     struct strbuf data, ctl;
                    671:     dl_hp_rawdata_req_t *rdata;
                    672:     int c;                     
                    673:                           
                    674:     if (l == NULL)
                    675:     {             
                    676:         return (-1);           
                    677:     }       
                    678:             
                    679:     rdata = (dl_hp_rawdata_req_t *)ctlbuf;
                    680:     rdata->dl_primitive = DL_HP_RAWDATA_REQ; 
                    681: 
                    682:     /* send it */                                  
                    683:     ctl.len = sizeof(dl_hp_rawdata_req_t);                         
                    684:     ctl.maxlen = sizeof(dl_hp_rawdata_req_t);                      
                    685:     ctl.buf = (int8_t *)rdata;
                    686: 
                    687:     data.maxlen = size;
                    688:     data.len    = size;                                
                    689:     data.buf    = packet;  
                    690: 
                    691:     c = putmsg(l->fd, &ctl, &data, 0);
                    692:     if (c == -1)                      
                    693:     {                    
                    694:         snprintf(l->err_buf, LIBNET_ERRBUF_SIZE,
                    695:                 "libnet_write_link(): %d bytes written (%s)\n", c,
                    696:                 strerror(errno));
                    697:         return (-1);
                    698:     }
                    699:     else
                    700:     {
                    701:         return (size);
                    702:     }                
                    703: }   
                    704: #else
                    705: int
1.1.1.2 ! misho     706: libnet_write_link(libnet_t *l, const uint8_t *packet, uint32_t size)
1.1       misho     707: {
                    708:     int c;
                    709:     struct strbuf data;
                    710: 
                    711:     if (l == NULL)
                    712:     { 
                    713:         return (-1);
                    714:     } 
                    715: 
                    716:     data.maxlen = size;
                    717:     data.len    = size;
                    718:     data.buf    = packet;
                    719: 
                    720:     c = putmsg(l->fd, NULL, &data, 0);
                    721:     if (c == -1)
                    722:     {
                    723:         snprintf(l->err_buf, LIBNET_ERRBUF_SIZE,
                    724:                 "libnet_write_link: %d bytes written (%s)\n", c,
                    725:                 strerror(errno));
                    726:         return (-1);
                    727:     }
                    728:     else
                    729:     {
                    730:         return (size);
                    731:     }
                    732: }
                    733: #endif
                    734: 
                    735: struct libnet_ether_addr *
                    736: libnet_get_hwaddr(libnet_t *l)
                    737: {
1.1.1.2 ! misho     738:     /* This implementation is not-reentrant. */
        !           739:     static int8_t buf[2048];
1.1       misho     740:     union DL_primitives *dlp;
                    741:     struct libnet_ether_addr *eap;
                    742: 
                    743:     if (l == NULL)
                    744:     { 
                    745:         return (NULL);
                    746:     }
                    747: 
                    748:     dlp = (union DL_primitives *)buf;
                    749: 
                    750:     dlp->physaddr_req.dl_primitive = DL_PHYS_ADDR_REQ;
                    751:     dlp->physaddr_req.dl_addr_type = DL_CURR_PHYS_ADDR;
                    752: 
                    753:     if (send_request(l->fd, (int8_t *)dlp, DL_PHYS_ADDR_REQ_SIZE, "physaddr",
                    754:             l->err_buf, 0) < 0)
                    755:     {
                    756:         return (NULL);
                    757:     }
                    758:     if (recv_ack(l->fd, DL_PHYS_ADDR_ACK_SIZE, "physaddr", (int8_t *)dlp,
                    759:             l->err_buf) < 0)
                    760:     {
                    761:         return (NULL);
                    762:     }
                    763: 
                    764:     eap = (struct libnet_ether_addr *)
                    765:             ((int8_t *) dlp + dlp->physaddr_ack.dl_addr_offset);
                    766:     return (eap);
                    767: }   
                    768: 
                    769: /* EOF */

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>