Annotation of embedaddon/libnet/src/libnet_build_bgp.c, revision 1.1
1.1 ! misho 1: /*
! 2: * $Id: libnet_build_bgp.c,v 1.7 2003/10/30 23:26:32 mike Exp $
! 3: *
! 4: * libnet
! 5: * libnet_build_bgp.c - BGP packet assembler (RFC 1771)
! 6: *
! 7: * Copyright (c) 2003 Frederic Raynal <pappy@security-labs.org>
! 8: * All rights reserved.
! 9: *
! 10: * Redistribution and use in source and binary forms, with or without
! 11: * modification, are permitted provided that the following conditions
! 12: * are met:
! 13: * 1. Redistributions of source code must retain the above copyright
! 14: * notice, this list of conditions and the following disclaimer.
! 15: * 2. Redistributions in binary form must reproduce the above copyright
! 16: * notice, this list of conditions and the following disclaimer in the
! 17: * documentation and/or other materials provided with the distribution.
! 18: *
! 19: * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
! 20: * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
! 21: * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
! 22: * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
! 23: * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
! 24: * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
! 25: * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
! 26: * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
! 27: * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
! 28: * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
! 29: * SUCH DAMAGE.
! 30: *
! 31: */
! 32:
! 33: #if (HAVE_CONFIG_H)
! 34: #include "../include/config.h"
! 35: #endif
! 36: #if (!(_WIN32) || (__CYGWIN__))
! 37: #include "../include/libnet.h"
! 38: #else
! 39: #include "../include/win32/libnet.h"
! 40: #endif
! 41:
! 42: libnet_ptag_t
! 43: libnet_build_bgp4_header(u_int8_t marker[LIBNET_BGP4_MARKER_SIZE],
! 44: u_int16_t len, u_int8_t type, u_int8_t *payload, u_int32_t payload_s,
! 45: libnet_t *l, libnet_ptag_t ptag)
! 46: {
! 47: u_int32_t n, h;
! 48: libnet_pblock_t *p;
! 49: struct libnet_bgp4_header_hdr bgp4_hdr;
! 50:
! 51: if (l == NULL)
! 52: {
! 53: return (-1);
! 54: }
! 55:
! 56: n = LIBNET_BGP4_HEADER_H + payload_s; /* size of memory block */
! 57: h = 0; /* BGP headers have no checksum */
! 58:
! 59: /*
! 60: * Find the existing protocol block if a ptag is specified, or create
! 61: * a new one.
! 62: */
! 63: p = libnet_pblock_probe(l, ptag, n, LIBNET_PBLOCK_BGP4_HEADER_H);
! 64: if (p == NULL)
! 65: {
! 66: return (-1);
! 67: }
! 68:
! 69: memset(&bgp4_hdr, 0, sizeof(bgp4_hdr));
! 70: memcpy(bgp4_hdr.marker, marker, LIBNET_BGP4_MARKER_SIZE * sizeof (u_int8_t));
! 71: bgp4_hdr.len = htons(len);
! 72: bgp4_hdr.type = type;
! 73:
! 74: n = libnet_pblock_append(l, p, (u_int8_t *)&bgp4_hdr, LIBNET_BGP4_HEADER_H);
! 75: if (n == -1)
! 76: {
! 77: goto bad;
! 78: }
! 79:
! 80: if ((payload && !payload_s) || (!payload && payload_s))
! 81: {
! 82: snprintf(l->err_buf, LIBNET_ERRBUF_SIZE,
! 83: "%s(): payload inconsistency\n", __func__);
! 84: goto bad;
! 85: }
! 86:
! 87: if (payload && payload_s)
! 88: {
! 89: n = libnet_pblock_append(l, p, payload, payload_s);
! 90: if (n == -1)
! 91: {
! 92: /* err msg set in libnet_pblock_append() */
! 93: goto bad;
! 94: }
! 95: }
! 96:
! 97: return (ptag ? ptag : libnet_pblock_update(l, p, h,
! 98: LIBNET_PBLOCK_BGP4_HEADER_H));
! 99: bad:
! 100: libnet_pblock_delete(l, p);
! 101: return (-1);
! 102: }
! 103:
! 104: libnet_ptag_t
! 105: libnet_build_bgp4_open(u_int8_t version, u_int16_t src_as, u_int16_t hold_time,
! 106: u_int32_t bgp_id, u_int8_t opt_len, u_int8_t *payload, u_int32_t payload_s,
! 107: libnet_t *l, libnet_ptag_t ptag)
! 108: {
! 109: u_int32_t n, h;
! 110: libnet_pblock_t *p;
! 111: u_int16_t val;
! 112:
! 113: if (l == NULL)
! 114: {
! 115: return (-1);
! 116: }
! 117:
! 118: n = LIBNET_BGP4_OPEN_H + payload_s; /* size of memory block */
! 119: h = 0; /* BGP msg have no checksum */
! 120:
! 121: /*
! 122: * Find the existing protocol block if a ptag is specified, or create
! 123: * a new one.
! 124: */
! 125: p = libnet_pblock_probe(l, ptag, n, LIBNET_PBLOCK_BGP4_OPEN_H);
! 126: if (p == NULL)
! 127: {
! 128: return (-1);
! 129: }
! 130:
! 131: /* for memory alignment reason, we need to append each field separately */
! 132: n = libnet_pblock_append(l, p, (u_int8_t *)&version, sizeof (version));
! 133: if (n == -1)
! 134: {
! 135: goto bad;
! 136: }
! 137:
! 138: val = htons(src_as);
! 139: n = libnet_pblock_append(l, p, (u_int8_t *)&val, sizeof(src_as));
! 140: if (n == -1)
! 141: {
! 142: goto bad;
! 143: }
! 144:
! 145: val = htons(hold_time);
! 146: n = libnet_pblock_append(l, p, (u_int8_t *)&val, sizeof(hold_time));
! 147: if (n == -1)
! 148: {
! 149: goto bad;
! 150: }
! 151:
! 152: n = htonl(bgp_id);
! 153: n = libnet_pblock_append(l, p, (u_int8_t *)&n, sizeof(bgp_id));
! 154: if (n == -1)
! 155: {
! 156: goto bad;
! 157: }
! 158:
! 159: n = libnet_pblock_append(l, p, (u_int8_t *)&opt_len, sizeof(opt_len));
! 160: if (n == -1)
! 161: {
! 162: goto bad;
! 163: }
! 164:
! 165: if ((payload && !payload_s) || (!payload && payload_s))
! 166: {
! 167: snprintf(l->err_buf, LIBNET_ERRBUF_SIZE,
! 168: "%s(): payload inconsistency\n", __func__);
! 169: goto bad;
! 170: }
! 171:
! 172: if (payload && payload_s)
! 173: {
! 174: n = libnet_pblock_append(l, p, payload, payload_s);
! 175: if (n == -1)
! 176: {
! 177: /* err msg set in libnet_pblock_append() */
! 178: goto bad;
! 179: }
! 180: }
! 181:
! 182: return (ptag ? ptag : libnet_pblock_update(l, p, h,
! 183: LIBNET_PBLOCK_BGP4_OPEN_H));
! 184: bad:
! 185: libnet_pblock_delete(l, p);
! 186: return (-1);
! 187: }
! 188:
! 189: libnet_ptag_t
! 190: libnet_build_bgp4_update(u_int16_t unfeasible_rt_len, u_int8_t *withdrawn_rt,
! 191: u_int16_t total_path_attr_len, u_int8_t *path_attributes,
! 192: u_int16_t info_len, u_int8_t *reachability_info,
! 193: u_int8_t *payload, u_int32_t payload_s, libnet_t *l, libnet_ptag_t ptag)
! 194: {
! 195: u_int32_t n, h;
! 196: libnet_pblock_t *p;
! 197: u_int16_t length;
! 198:
! 199: if (l == NULL)
! 200: {
! 201: return (-1);
! 202: }
! 203:
! 204: /* size of memory block */
! 205: n = LIBNET_BGP4_UPDATE_H + unfeasible_rt_len + total_path_attr_len +
! 206: info_len + payload_s;
! 207:
! 208: /* BGP msg have no checksum */
! 209: h = 0;
! 210:
! 211: /*
! 212: * Find the existing protocol block if a ptag is specified, or create
! 213: * a new one.
! 214: */
! 215: p = libnet_pblock_probe(l, ptag, n, LIBNET_PBLOCK_BGP4_UPDATE_H);
! 216: if (p == NULL)
! 217: {
! 218: return (-1);
! 219: }
! 220:
! 221: /* for memory alignment reason, we need to append each field separately */
! 222: length = htons(unfeasible_rt_len);
! 223: n = libnet_pblock_append(l, p, (u_int8_t *)&length,
! 224: sizeof (unfeasible_rt_len));
! 225: if (n == -1)
! 226: {
! 227: goto bad;
! 228: }
! 229:
! 230: if (unfeasible_rt_len && withdrawn_rt)
! 231: {
! 232: n = libnet_pblock_append(l, p, withdrawn_rt, unfeasible_rt_len);
! 233: if (n == -1)
! 234: {
! 235: goto bad;
! 236: }
! 237: }
! 238:
! 239: length = htons(total_path_attr_len);
! 240: n = libnet_pblock_append(l, p, (u_int8_t *)&length,
! 241: sizeof (total_path_attr_len));
! 242: if (n == -1)
! 243: {
! 244: goto bad;
! 245: }
! 246:
! 247: if (total_path_attr_len && path_attributes)
! 248: {
! 249: n = libnet_pblock_append(l, p, path_attributes, total_path_attr_len);
! 250: if (n == -1)
! 251: {
! 252: goto bad;
! 253: }
! 254: }
! 255:
! 256: if (info_len && reachability_info)
! 257: {
! 258: n = libnet_pblock_append(l, p, reachability_info, info_len);
! 259: if (n == -1)
! 260: {
! 261: goto bad;
! 262: }
! 263: }
! 264:
! 265: if ((payload && !payload_s) || (!payload && payload_s))
! 266: {
! 267: snprintf(l->err_buf, LIBNET_ERRBUF_SIZE,
! 268: "%s(): payload inconsistency\n", __func__);
! 269: goto bad;
! 270: }
! 271:
! 272: if (payload && payload_s)
! 273: {
! 274: n = libnet_pblock_append(l, p, payload, payload_s);
! 275: if (n == -1)
! 276: {
! 277: /* err msg set in libnet_pblock_append() */
! 278: goto bad;
! 279: }
! 280: }
! 281:
! 282: return (ptag ? ptag : libnet_pblock_update(l, p, h,
! 283: LIBNET_PBLOCK_BGP4_UPDATE_H));
! 284: bad:
! 285: libnet_pblock_delete(l, p);
! 286: return (-1);
! 287: }
! 288:
! 289: libnet_ptag_t
! 290: libnet_build_bgp4_notification(u_int8_t err_code, u_int8_t err_subcode,
! 291: u_int8_t *payload, u_int32_t payload_s, libnet_t *l, libnet_ptag_t ptag)
! 292: {
! 293: u_int32_t n, h;
! 294: libnet_pblock_t *p;
! 295: struct libnet_bgp4_notification_hdr bgp4_hdr;
! 296:
! 297: if (l == NULL)
! 298: {
! 299: return (-1);
! 300: }
! 301:
! 302: n = LIBNET_BGP4_NOTIFICATION_H + + payload_s; /* size of memory block */
! 303: h = 0;
! 304:
! 305: /*
! 306: * Find the existing protocol block if a ptag is specified, or create
! 307: * a new one.
! 308: */
! 309: p = libnet_pblock_probe(l, ptag, n, LIBNET_PBLOCK_BGP4_NOTIFICATION_H);
! 310: if (p == NULL)
! 311: {
! 312: return (-1);
! 313: }
! 314:
! 315: memset(&bgp4_hdr, 0, sizeof(bgp4_hdr));
! 316: bgp4_hdr.err_code = err_code;
! 317: bgp4_hdr.err_subcode = err_subcode;
! 318:
! 319: n = libnet_pblock_append(l, p, (u_int8_t *)&bgp4_hdr,
! 320: LIBNET_BGP4_NOTIFICATION_H);
! 321: if (n == -1)
! 322: {
! 323: goto bad;
! 324: }
! 325:
! 326: if ((payload && !payload_s) || (!payload && payload_s))
! 327: {
! 328: snprintf(l->err_buf, LIBNET_ERRBUF_SIZE,
! 329: "%s(): payload inconsistency\n", __func__);
! 330: goto bad;
! 331: }
! 332:
! 333: if (payload && payload_s)
! 334: {
! 335: n = libnet_pblock_append(l, p, payload, payload_s);
! 336: if (n == -1)
! 337: {
! 338: /* err msg set in libnet_pblock_append() */
! 339: goto bad;
! 340: }
! 341: }
! 342:
! 343: return (ptag ? ptag : libnet_pblock_update(l, p, h,
! 344: LIBNET_PBLOCK_BGP4_NOTIFICATION_H));
! 345: bad:
! 346: libnet_pblock_delete(l, p);
! 347: return (-1);
! 348: }
! 349:
! 350: /* EOF */
FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>